mirror of
https://github.com/rectorphp/rector.git
synced 2025-02-24 03:35:01 +01:00
147 lines
5.8 KiB
PHP
147 lines
5.8 KiB
PHP
<?php
|
|
|
|
declare (strict_types=1);
|
|
namespace Rector\Php71\NodeAnalyzer;
|
|
|
|
use PhpParser\Node\Expr;
|
|
use PhpParser\Node\Expr\Array_;
|
|
use PhpParser\Node\Expr\PropertyFetch;
|
|
use PhpParser\Node\Expr\StaticPropertyFetch;
|
|
use PhpParser\Node\Stmt;
|
|
use PhpParser\Node\Stmt\ClassLike;
|
|
use PHPStan\Analyser\Scope;
|
|
use PHPStan\Reflection\ClassReflection;
|
|
use PHPStan\Reflection\Php\PhpPropertyReflection;
|
|
use PHPStan\Reflection\PropertyReflection;
|
|
use PHPStan\Reflection\ReflectionProvider;
|
|
use PHPStan\Type\ArrayType;
|
|
use PHPStan\Type\Constant\ConstantArrayType;
|
|
use PHPStan\Type\Type;
|
|
use PHPStan\Type\TypeWithClassName;
|
|
use PHPStan\Type\UnionType;
|
|
use Rector\Core\NodeAnalyzer\PropertyFetchAnalyzer;
|
|
use Rector\Core\PhpParser\Node\BetterNodeFinder;
|
|
use Rector\NodeNameResolver\NodeNameResolver;
|
|
use Rector\NodeTypeResolver\Node\AttributeKey;
|
|
use Rector\NodeTypeResolver\NodeTypeResolver;
|
|
use Rector\TypeDeclaration\AlreadyAssignDetector\ConstructorAssignDetector;
|
|
final class CountableAnalyzer
|
|
{
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\NodeTypeResolver\NodeTypeResolver
|
|
*/
|
|
private $nodeTypeResolver;
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\NodeNameResolver\NodeNameResolver
|
|
*/
|
|
private $nodeNameResolver;
|
|
/**
|
|
* @readonly
|
|
* @var \PHPStan\Reflection\ReflectionProvider
|
|
*/
|
|
private $reflectionProvider;
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\Core\PhpParser\Node\BetterNodeFinder
|
|
*/
|
|
private $betterNodeFinder;
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\Core\NodeAnalyzer\PropertyFetchAnalyzer
|
|
*/
|
|
private $propertyFetchAnalyzer;
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\TypeDeclaration\AlreadyAssignDetector\ConstructorAssignDetector
|
|
*/
|
|
private $constructorAssignDetector;
|
|
public function __construct(NodeTypeResolver $nodeTypeResolver, NodeNameResolver $nodeNameResolver, ReflectionProvider $reflectionProvider, BetterNodeFinder $betterNodeFinder, PropertyFetchAnalyzer $propertyFetchAnalyzer, ConstructorAssignDetector $constructorAssignDetector)
|
|
{
|
|
$this->nodeTypeResolver = $nodeTypeResolver;
|
|
$this->nodeNameResolver = $nodeNameResolver;
|
|
$this->reflectionProvider = $reflectionProvider;
|
|
$this->betterNodeFinder = $betterNodeFinder;
|
|
$this->propertyFetchAnalyzer = $propertyFetchAnalyzer;
|
|
$this->constructorAssignDetector = $constructorAssignDetector;
|
|
}
|
|
public function isCastableArrayType(Expr $expr, ArrayType $arrayType) : bool
|
|
{
|
|
if (!$this->propertyFetchAnalyzer->isPropertyFetch($expr)) {
|
|
return \false;
|
|
}
|
|
if ($arrayType instanceof ConstantArrayType) {
|
|
return \false;
|
|
}
|
|
/** @var StaticPropertyFetch|PropertyFetch $expr */
|
|
$callerObjectType = $expr instanceof StaticPropertyFetch ? $this->nodeTypeResolver->getType($expr->class) : $this->nodeTypeResolver->getType($expr->var);
|
|
$propertyName = $this->nodeNameResolver->getName($expr->name);
|
|
if (!\is_string($propertyName)) {
|
|
return \false;
|
|
}
|
|
if ($callerObjectType instanceof UnionType) {
|
|
$callerObjectType = $callerObjectType->getTypes()[0];
|
|
}
|
|
if (!$callerObjectType instanceof TypeWithClassName) {
|
|
return \false;
|
|
}
|
|
if ($this->isCallerObjectClassNameStmtOrArray($callerObjectType)) {
|
|
return \false;
|
|
}
|
|
// this must be handled reflection, as PHPStan ReflectionProvider does not provide default values for properties in any way
|
|
$classReflection = $this->reflectionProvider->getClass($callerObjectType->getClassName());
|
|
$nativeReflectionClass = $classReflection->getNativeReflection();
|
|
$propertiesDefaults = $nativeReflectionClass->getDefaultProperties();
|
|
if (!\array_key_exists($propertyName, $propertiesDefaults)) {
|
|
return \false;
|
|
}
|
|
$phpPropertyReflection = $this->resolveProperty($expr, $classReflection, $propertyName);
|
|
if (!$phpPropertyReflection instanceof PhpPropertyReflection) {
|
|
return \false;
|
|
}
|
|
$nativeType = $phpPropertyReflection->getNativeType();
|
|
if ($this->isIterableOrFilledAtConstruct($nativeType, $expr)) {
|
|
return \false;
|
|
}
|
|
$propertyDefaultValue = $propertiesDefaults[$propertyName];
|
|
return $propertyDefaultValue === null;
|
|
}
|
|
private function isCallerObjectClassNameStmtOrArray(TypeWithClassName $typeWithClassName) : bool
|
|
{
|
|
if (\is_a($typeWithClassName->getClassName(), Stmt::class, \true)) {
|
|
return \true;
|
|
}
|
|
return \is_a($typeWithClassName->getClassName(), Array_::class, \true);
|
|
}
|
|
/**
|
|
* @param \PhpParser\Node\Expr\StaticPropertyFetch|\PhpParser\Node\Expr\PropertyFetch $propertyFetch
|
|
*/
|
|
private function isIterableOrFilledAtConstruct(Type $nativeType, $propertyFetch) : bool
|
|
{
|
|
if ($nativeType->isIterable()->yes()) {
|
|
return \true;
|
|
}
|
|
$classLike = $this->betterNodeFinder->findParentType($propertyFetch, ClassLike::class);
|
|
if (!$classLike instanceof ClassLike) {
|
|
return \false;
|
|
}
|
|
if ($propertyFetch->name instanceof Expr) {
|
|
return \false;
|
|
}
|
|
$propertyName = (string) $this->nodeNameResolver->getName($propertyFetch->name);
|
|
return $this->constructorAssignDetector->isPropertyAssigned($classLike, $propertyName);
|
|
}
|
|
/**
|
|
* @param \PhpParser\Node\Expr\StaticPropertyFetch|\PhpParser\Node\Expr\PropertyFetch $propertyFetch
|
|
*/
|
|
private function resolveProperty($propertyFetch, ClassReflection $classReflection, string $propertyName) : ?PropertyReflection
|
|
{
|
|
$scope = $propertyFetch->getAttribute(AttributeKey::SCOPE);
|
|
if (!$scope instanceof Scope) {
|
|
return null;
|
|
}
|
|
return $classReflection->getProperty($propertyName, $scope);
|
|
}
|
|
}
|