rector/packages/StaticTypeMapper/Naming/NameScopeFactory.php

106 lines
4.3 KiB
PHP
Raw Normal View History

<?php
declare (strict_types=1);
namespace Rector\StaticTypeMapper\Naming;
use PhpParser\Node;
2021-01-08 22:30:33 +00:00
use PhpParser\Node\Stmt\ClassLike;
use PhpParser\Node\Stmt\Use_;
use PhpParser\Node\Stmt\UseUse;
use PHPStan\Analyser\NameScope;
use PHPStan\Analyser\Scope;
use PHPStan\Type\Generic\TemplateTypeMap;
2021-01-08 22:30:33 +00:00
use PHPStan\Type\Type;
use Rector\BetterPhpDocParser\PhpDocInfo\PhpDocInfoFactory;
use Rector\Core\Exception\ShouldNotHappenException;
use Rector\NodeTypeResolver\Node\AttributeKey;
use Rector\StaticTypeMapper\StaticTypeMapper;
/**
* @see https://github.com/phpstan/phpstan-src/blob/8376548f76e2c845ae047e3010e873015b796818/src/Analyser/NameScope.php#L32
*/
final class NameScopeFactory
{
/**
* @var StaticTypeMapper
*/
private $staticTypeMapper;
/**
* @var PhpDocInfoFactory
*/
private $phpDocInfoFactory;
/**
* This is needed to avoid circular references
* @required
*/
public function autowireNameScopeFactory(\Rector\BetterPhpDocParser\PhpDocInfo\PhpDocInfoFactory $phpDocInfoFactory) : void
{
$this->phpDocInfoFactory = $phpDocInfoFactory;
}
public function createNameScopeFromNodeWithoutTemplateTypes(\PhpParser\Node $node) : \PHPStan\Analyser\NameScope
{
$scope = $node->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::SCOPE);
$namespace = $scope instanceof \PHPStan\Analyser\Scope ? $scope->getNamespace() : null;
/** @var Use_[] $useNodes */
$useNodes = (array) $node->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::USE_NODES);
$uses = $this->resolveUseNamesByAlias($useNodes);
$className = $node->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::CLASS_NAME);
return new \PHPStan\Analyser\NameScope($namespace, $uses, $className);
}
public function createNameScopeFromNode(\PhpParser\Node $node) : \PHPStan\Analyser\NameScope
{
$nameScope = $this->createNameScopeFromNodeWithoutTemplateTypes($node);
$templateTypeMap = $this->templateTemplateTypeMap($node);
return new \PHPStan\Analyser\NameScope($nameScope->getNamespace(), $nameScope->getUses(), $nameScope->getClassName(), null, $templateTypeMap);
}
public function setStaticTypeMapper(\Rector\StaticTypeMapper\StaticTypeMapper $staticTypeMapper) : void
{
$this->staticTypeMapper = $staticTypeMapper;
}
/**
* @param Use_[] $useNodes
* @return array<string, string>
*/
private function resolveUseNamesByAlias(array $useNodes) : array
{
$useNamesByAlias = [];
foreach ($useNodes as $useNode) {
foreach ($useNode->uses as $useUse) {
/** @var UseUse $useUse */
$aliasName = $useUse->getAlias()->name;
$useName = $useUse->name->toString();
if (!\is_string($useName)) {
throw new \Rector\Core\Exception\ShouldNotHappenException();
}
// uses must be lowercase, as PHPStan lowercases it
$lowercasedAliasName = \strtolower($aliasName);
$useNamesByAlias[$lowercasedAliasName] = $useName;
}
}
return $useNamesByAlias;
}
private function templateTemplateTypeMap(\PhpParser\Node $node) : \PHPStan\Type\Generic\TemplateTypeMap
2021-01-08 22:30:33 +00:00
{
$nodeTemplateTypes = $this->resolveTemplateTypesFromNode($node);
$class = $node->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::CLASS_NODE);
2021-01-08 22:30:33 +00:00
$classTemplateTypes = [];
if ($class instanceof \PhpParser\Node\Stmt\ClassLike) {
2021-01-08 22:30:33 +00:00
$classTemplateTypes = $this->resolveTemplateTypesFromNode($class);
}
$templateTypes = \array_merge($nodeTemplateTypes, $classTemplateTypes);
return new \PHPStan\Type\Generic\TemplateTypeMap($templateTypes);
2021-01-08 22:30:33 +00:00
}
/**
* @return Type[]
*/
private function resolveTemplateTypesFromNode(\PhpParser\Node $node) : array
{
$phpDocInfo = $this->phpDocInfoFactory->createFromNodeOrEmpty($node);
$templateTypes = [];
2021-01-08 22:30:33 +00:00
foreach ($phpDocInfo->getTemplateTagValueNodes() as $templateTagValueNode) {
$phpstanType = $this->staticTypeMapper->mapPHPStanPhpDocTypeToPHPStanType($templateTagValueNode, $node);
$templateTypes[$templateTagValueNode->name] = $phpstanType;
}
2021-01-08 22:30:33 +00:00
return $templateTypes;
}
}