rector/src/PHPStanStaticTypeMapper/TypeMapper/IntersectionTypeMapper.php
Tomas Votruba e08ae92529 Updated Rector to commit fb312e00c38b17b096a7257151a4ec0571cd8607
fb312e00c3 [PHPStanStaticTypeMapper] Avoid double \ prefix on IntersectionTypeMapper (#5656)
2024-02-22 10:08:06 +00:00

96 lines
3.4 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\PHPStanStaticTypeMapper\TypeMapper;
use PhpParser\Node;
use PhpParser\Node\Name\FullyQualified;
use PHPStan\PhpDocParser\Ast\Node as AstNode;
use PHPStan\PhpDocParser\Ast\Type\IdentifierTypeNode;
use PHPStan\PhpDocParser\Ast\Type\TypeNode;
use PHPStan\Type\IntersectionType;
use PHPStan\Type\ObjectType;
use PHPStan\Type\ObjectWithoutClassType;
use PHPStan\Type\Type;
use Rector\Php\PhpVersionProvider;
use Rector\PhpDocParser\PhpDocParser\PhpDocNodeTraverser;
use Rector\PHPStanStaticTypeMapper\Contract\TypeMapperInterface;
use Rector\ValueObject\PhpVersionFeature;
/**
* @implements TypeMapperInterface<IntersectionType>
*/
final class IntersectionTypeMapper implements TypeMapperInterface
{
/**
* @readonly
* @var \Rector\Php\PhpVersionProvider
*/
private $phpVersionProvider;
/**
* @readonly
* @var \Rector\PHPStanStaticTypeMapper\TypeMapper\ObjectWithoutClassTypeMapper
*/
private $objectWithoutClassTypeMapper;
/**
* @readonly
* @var \Rector\PHPStanStaticTypeMapper\TypeMapper\ObjectTypeMapper
*/
private $objectTypeMapper;
public function __construct(PhpVersionProvider $phpVersionProvider, \Rector\PHPStanStaticTypeMapper\TypeMapper\ObjectWithoutClassTypeMapper $objectWithoutClassTypeMapper, \Rector\PHPStanStaticTypeMapper\TypeMapper\ObjectTypeMapper $objectTypeMapper)
{
$this->phpVersionProvider = $phpVersionProvider;
$this->objectWithoutClassTypeMapper = $objectWithoutClassTypeMapper;
$this->objectTypeMapper = $objectTypeMapper;
}
public function getNodeClass() : string
{
return IntersectionType::class;
}
/**
* @param IntersectionType $type
*/
public function mapToPHPStanPhpDocTypeNode(Type $type) : TypeNode
{
$typeNode = $type->toPhpDocNode();
$phpDocNodeTraverser = new PhpDocNodeTraverser();
$phpDocNodeTraverser->traverseWithCallable($typeNode, '', static function (AstNode $astNode) : ?IdentifierTypeNode {
if ($astNode instanceof IdentifierTypeNode) {
$astNode->name = '\\' . \ltrim($astNode->name, '\\');
return $astNode;
}
return null;
});
return $typeNode;
}
/**
* @param IntersectionType $type
*/
public function mapToPhpParserNode(Type $type, string $typeKind) : ?Node
{
if (!$this->phpVersionProvider->isAtLeastPhpVersion(PhpVersionFeature::INTERSECTION_TYPES)) {
return null;
}
$intersectionedTypeNodes = [];
foreach ($type->getTypes() as $type) {
if ($type instanceof ObjectWithoutClassType) {
return $this->objectWithoutClassTypeMapper->mapToPhpParserNode($type, $typeKind);
}
if (!$type instanceof ObjectType) {
return null;
}
$resolvedType = $this->objectTypeMapper->mapToPhpParserNode($type, $typeKind);
if (!$resolvedType instanceof FullyQualified) {
return null;
}
$intersectionedTypeNodes[] = $resolvedType;
}
if ($intersectionedTypeNodes === []) {
return null;
}
if (\count($intersectionedTypeNodes) === 1) {
return \current($intersectionedTypeNodes);
}
return new Node\IntersectionType($intersectionedTypeNodes);
}
}