Valinor/tests/Unit/Utility/Reflection/ReflectionTest.php
Maximilian Bösing 64e0a2d5ac
feat: allow psalm and phpstan prefix in docblocks
The following annotations are now properly handled: `@psalm-param`, 
`@phpstan-param`, `@psalm-return` and `@phpstan-return`.

If one of those found along with a basic `@param` or `@return` 
annotation, it will override the basic value.
2022-05-05 19:40:11 +02:00

425 lines
12 KiB
PHP

<?php
declare(strict_types=1);
namespace CuyZ\Valinor\Tests\Unit\Utility\Reflection;
use Closure;
use CuyZ\Valinor\Tests\Fake\FakeReflector;
use CuyZ\Valinor\Tests\Fixture\Object\ObjectWithPropertyWithNativeIntersectionType;
use CuyZ\Valinor\Tests\Fixture\Object\ObjectWithPropertyWithNativeUnionType;
use CuyZ\Valinor\Utility\Reflection\Reflection;
use PHPUnit\Framework\TestCase;
use ReflectionClass;
use ReflectionFunction;
use ReflectionParameter;
use ReflectionProperty;
use ReflectionType;
use Reflector;
use RuntimeException;
use stdClass;
use function get_class;
final class ReflectionTest extends TestCase
{
public function test_get_class_reflection_returns_class_reflection(): void
{
$className = stdClass::class;
$classReflection = Reflection::class($className);
self::assertSame($className, $classReflection->getName());
}
public function test_class_reflection_is_created_only_once(): void
{
$className = stdClass::class;
$classReflectionA = Reflection::class($className);
$classReflectionB = Reflection::class($className);
self::assertSame($classReflectionA, $classReflectionB);
}
public function test_function_reflection_is_created_only_once(): void
{
$function = fn () => 42;
$functionReflectionA = Reflection::function($function);
$functionReflectionB = Reflection::function($function);
self::assertSame($functionReflectionA, $functionReflectionB);
}
public function test_reflection_signatures_are_correct(): void
{
$class = get_class(new class () {
public string $property;
public function method(string $parameter): void
{
}
});
$reflectionClass = new ReflectionClass($class);
$reflectionProperty = $reflectionClass->getProperty('property');
$reflectionMethod = $reflectionClass->getMethod('method');
$reflectionParameter = $reflectionMethod->getParameters()[0];
self::assertSame($class, Reflection::signature($reflectionClass));
self::assertSame($class . '::$property', Reflection::signature($reflectionProperty));
self::assertSame($class . '::method()', Reflection::signature($reflectionMethod));
self::assertSame($class . '::method($parameter)', Reflection::signature($reflectionParameter));
}
public function test_invalid_reflection_signature_throws_exception(): void
{
$this->expectException(RuntimeException::class);
$this->expectExceptionMessage('Invalid reflection type `' . FakeReflector::class . '`.');
$wrongReflector = new FakeReflector();
Reflection::signature($wrongReflector);
}
public function test_scalar_type_is_handled(): void
{
$object = new class () {
public string $someProperty;
};
/** @var ReflectionType $type */
$type = (new ReflectionProperty($object, 'someProperty'))->getType();
self::assertSame('string', Reflection::flattenType($type));
}
public function test_nullable_scalar_type_is_handled(): void
{
$object = new class () {
public ?string $someProperty;
};
/** @var ReflectionType $type */
$type = (new ReflectionProperty($object, 'someProperty'))->getType();
self::assertSame('string|null', Reflection::flattenType($type));
}
/**
* @requires PHP >= 8
*/
public function test_union_type_is_handled(): void
{
$class = ObjectWithPropertyWithNativeUnionType::class;
/** @var ReflectionType $type */
$type = (new ReflectionProperty($class, 'someProperty'))->getType();
self::assertSame('int|float', Reflection::flattenType($type));
}
/**
* @requires PHP >= 8
*/
public function test_mixed_type_is_handled(): void
{
$object = new class () {
public mixed $someProperty;
};
/** @var ReflectionType $type */
$type = (new ReflectionProperty($object, 'someProperty'))->getType();
self::assertSame('mixed', Reflection::flattenType($type));
}
/**
* @requires PHP >= 8.1
*/
public function test_intersection_type_is_handled(): void
{
$class = ObjectWithPropertyWithNativeIntersectionType::class;
/** @var ReflectionType $type */
$type = (new ReflectionProperty($class, 'someProperty'))->getType();
self::assertSame('Countable&Iterator', Reflection::flattenType($type));
}
/**
* @param non-empty-string $expectedType
* @dataProvider callables_with_docblock_typed_return_type
*/
public function test_docblock_return_type_is_fetched_correctly(
callable $dockblockTypedCallable,
string $expectedType
): void {
$type = Reflection::docBlockReturnType(new ReflectionFunction(Closure::fromCallable($dockblockTypedCallable)));
self::assertSame($expectedType, $type);
}
public function test_docblock_return_type_with_no_docblock_returns_null(): void
{
$callable = static function (): void {
};
$type = Reflection::docBlockReturnType(new ReflectionFunction($callable));
self::assertNull($type);
}
/**
* @param ReflectionParameter|ReflectionProperty $property
* @param non-empty-string $expectedType
* @dataProvider objects_with_docblock_typed_properties
*/
public function test_docblock_var_type_is_fetched_correctly(
Reflector $property,
string $expectedType
): void {
self::assertEquals($expectedType, Reflection::docBlockType($property));
}
/**
* @return iterable<non-empty-string,array{0:callable,1:non-empty-string}>
*/
public function callables_with_docblock_typed_return_type(): iterable
{
yield 'phpdoc' => [
/** @return int */
fn () => 42,
'int',
];
yield 'phpdoc followed by new line' => [
/**
* @return int
*
*/
fn () => 42,
'int',
];
yield 'phpdoc literal string' => [
/** @return 'foo' */
fn () => 'foo',
'\'foo\'',
];
yield 'psalm' => [
/** @psalm-return int */
fn () => 42,
'int',
];
yield 'psalm trailing' => [
/**
* @return int
* @psalm-return positive-int
*/
fn () => 42,
'positive-int',
];
yield 'psalm leading' => [
/**
* @psalm-return positive-int
* @return int
*/
fn () => 42,
'positive-int',
];
yield 'phpstan' => [
/** @phpstan-return int */
fn () => 42,
'int',
];
yield 'phpstan trailing' => [
/**
* @return int
* @phpstan-return positive-int
*/
fn () => 42,
'positive-int',
];
yield 'phpstan leading' => [
/**
* @phpstan-return positive-int
* @return int
*/
fn () => 42,
'positive-int',
];
}
/**
* @return iterable<non-empty-string,array{0:ReflectionProperty|ReflectionParameter,1:non-empty-string}>
*/
public function objects_with_docblock_typed_properties(): iterable
{
yield 'phpdoc @var' => [
new ReflectionProperty(new class () {
/** @var string */
public $foo;
}, 'foo'),
'string',
];
yield 'phpdoc @var followed by new line' => [
new ReflectionProperty(new class () {
/**
* @var string
*
*/
public $foo;
}, 'foo'),
'string',
];
yield 'psalm @var standalone' => [
new ReflectionProperty(new class () {
/** @psalm-var string */
public $foo;
}, 'foo'),
'string',
];
yield 'psalm @var leading' => [
new ReflectionProperty(new class () {
/**
* @psalm-var non-empty-string
* @var string
*/
public $foo;
}, 'foo'),
'non-empty-string',
];
yield 'psalm @var trailing' => [
new ReflectionProperty(new class () {
/**
* @var string
* @psalm-var non-empty-string
*/
public $foo;
}, 'foo'),
'non-empty-string',
];
yield 'phpstan @var standalone' => [
new ReflectionProperty(new class () {
/** @phpstan-var string */
public $foo;
}, 'foo'),
'string',
];
yield 'phpstan @var leading' => [
new ReflectionProperty(new class () {
/**
* @phpstan-var non-empty-string
* @var string
*/
public $foo;
}, 'foo'),
'non-empty-string',
];
yield 'phpstan @var trailing' => [
new ReflectionProperty(new class () {
/**
* @var string
* @phpstan-var non-empty-string
*/
public $foo;
}, 'foo'),
'non-empty-string',
];
yield 'phpdoc @param' => [
new ReflectionParameter(
/** @param string $string */
static function ($string): void {
},
'string',
),
'string',
];
yield 'psalm @param standalone' => [
new ReflectionParameter(
/** @psalm-param string $string */
static function ($string): void {
},
'string',
),
'string',
];
yield 'psalm @param leading' => [
new ReflectionParameter(
/**
* @psalm-param non-empty-string $string
* @param string $string
*/
static function ($string): void {
},
'string',
),
'non-empty-string',
];
yield 'psalm @param trailing' => [
new ReflectionParameter(
/**
* @param string $string
* @psalm-param non-empty-string $string
*/
static function ($string): void {
},
'string',
),
'non-empty-string',
];
yield 'phpstan @param standalone' => [
new ReflectionParameter(
/** @phpstan-param string $string */
static function ($string): void {
},
'string',
),
'string',
];
yield 'phpstan @param leading' => [
new ReflectionParameter(
/**
* @phpstan-param non-empty-string $string
* @param string $string
*/
static function ($string): void {
},
'string',
),
'non-empty-string',
];
yield 'phpstan @param trailing' => [
new ReflectionParameter(
/**
* @param string $string
* @phpstan-param non-empty-string $string
*/
static function ($string): void {
},
'string',
),
'non-empty-string',
];
}
}