mirror of
https://github.com/rectorphp/rector.git
synced 2024-06-08 20:22:24 +00:00
6c20d257e3
d3d18d442c
keep relative paths
112 lines
4.0 KiB
PHP
112 lines
4.0 KiB
PHP
<?php
|
|
|
|
declare (strict_types=1);
|
|
namespace Rector\PHPStanStaticTypeMapper\TypeMapper;
|
|
|
|
use PhpParser\Node;
|
|
use PhpParser\Node\Name;
|
|
use PhpParser\Node\Name\FullyQualified;
|
|
use PHPStan\PhpDocParser\Ast\Type\TypeNode;
|
|
use PHPStan\Type\Generic\GenericClassStringType;
|
|
use PHPStan\Type\IntersectionType;
|
|
use PHPStan\Type\ObjectType;
|
|
use PHPStan\Type\Type;
|
|
use Rector\BetterPhpDocParser\ValueObject\Type\BracketsAwareIntersectionTypeNode;
|
|
use Rector\Core\Exception\ShouldNotHappenException;
|
|
use Rector\Core\Php\PhpVersionProvider;
|
|
use Rector\Core\ValueObject\PhpVersionFeature;
|
|
use Rector\PHPStanStaticTypeMapper\Contract\TypeMapperInterface;
|
|
use Rector\PHPStanStaticTypeMapper\PHPStanStaticTypeMapper;
|
|
use RectorPrefix202209\Symfony\Contracts\Service\Attribute\Required;
|
|
/**
|
|
* @implements TypeMapperInterface<IntersectionType>
|
|
*/
|
|
final class IntersectionTypeMapper implements TypeMapperInterface
|
|
{
|
|
/**
|
|
* @var string
|
|
*/
|
|
private const STRING = 'string';
|
|
/**
|
|
* @var \Rector\PHPStanStaticTypeMapper\PHPStanStaticTypeMapper
|
|
*/
|
|
private $phpStanStaticTypeMapper;
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\Core\Php\PhpVersionProvider
|
|
*/
|
|
private $phpVersionProvider;
|
|
public function __construct(PhpVersionProvider $phpVersionProvider)
|
|
{
|
|
$this->phpVersionProvider = $phpVersionProvider;
|
|
}
|
|
/**
|
|
* @required
|
|
*/
|
|
public function autowire(PHPStanStaticTypeMapper $phpStanStaticTypeMapper) : void
|
|
{
|
|
$this->phpStanStaticTypeMapper = $phpStanStaticTypeMapper;
|
|
}
|
|
/**
|
|
* @return class-string<Type>
|
|
*/
|
|
public function getNodeClass() : string
|
|
{
|
|
return IntersectionType::class;
|
|
}
|
|
/**
|
|
* @param IntersectionType $type
|
|
*/
|
|
public function mapToPHPStanPhpDocTypeNode(Type $type, string $typeKind) : TypeNode
|
|
{
|
|
$intersectionTypesNodes = [];
|
|
foreach ($type->getTypes() as $intersectionedType) {
|
|
$intersectionTypesNodes[] = $this->phpStanStaticTypeMapper->mapToPHPStanPhpDocTypeNode($intersectionedType, $typeKind);
|
|
}
|
|
$intersectionTypesNodes = \array_unique($intersectionTypesNodes);
|
|
if (\count($intersectionTypesNodes) === 1) {
|
|
return $intersectionTypesNodes[0];
|
|
}
|
|
return new BracketsAwareIntersectionTypeNode($intersectionTypesNodes);
|
|
}
|
|
/**
|
|
* @param IntersectionType $type
|
|
*/
|
|
public function mapToPhpParserNode(Type $type, string $typeKind) : ?Node
|
|
{
|
|
if (!$this->phpVersionProvider->isAtLeastPhpVersion(PhpVersionFeature::INTERSECTION_TYPES)) {
|
|
return $this->matchMockObjectType($type);
|
|
}
|
|
$intersectionedTypeNodes = [];
|
|
foreach ($type->getTypes() as $intersectionedType) {
|
|
$resolvedType = $this->phpStanStaticTypeMapper->mapToPhpParserNode($intersectionedType, $typeKind);
|
|
if ($intersectionedType instanceof GenericClassStringType) {
|
|
$resolvedTypeName = self::STRING;
|
|
$resolvedType = new Name(self::STRING);
|
|
} elseif (!$resolvedType instanceof Name) {
|
|
throw new ShouldNotHappenException();
|
|
} else {
|
|
$resolvedTypeName = (string) $resolvedType;
|
|
}
|
|
if (\in_array($resolvedTypeName, [self::STRING, 'object'], \true)) {
|
|
return $resolvedType;
|
|
}
|
|
$intersectionedTypeNodes[] = $resolvedType;
|
|
}
|
|
return new Node\IntersectionType($intersectionedTypeNodes);
|
|
}
|
|
private function matchMockObjectType(IntersectionType $intersectionType) : ?FullyQualified
|
|
{
|
|
// return mock object as the strict one
|
|
foreach ($intersectionType->getTypes() as $intersectionedType) {
|
|
if (!$intersectionedType instanceof ObjectType) {
|
|
continue;
|
|
}
|
|
if ($intersectionedType->getClassName() === 'PHPUnit\\Framework\\MockObject\\MockObject') {
|
|
return new FullyQualified($intersectionedType->getClassName());
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
}
|