* Dariusz Rumiński * * This source file is subject to the MIT license that is bundled * with this source code in the file LICENSE. */ namespace PhpCsFixer\Tests\DocBlock; use PhpCsFixer\DocBlock\TypeExpression; use PhpCsFixer\Tests\TestCase; use PhpCsFixer\Tokenizer\Analyzer\Analysis\NamespaceAnalysis; use PhpCsFixer\Tokenizer\Analyzer\Analysis\NamespaceUseAnalysis; /** * @covers \PhpCsFixer\DocBlock\TypeExpression * * @internal */ final class TypeExpressionTest extends TestCase { /** * @param null|string[] $expectedTypes * * @dataProvider provideGetConstTypesCases * @dataProvider provideGetTypesCases */ public function testGetTypes(string $typesExpression, array $expectedTypes = null): void { if (null === $expectedTypes) { $expectedTypes = [$typesExpression]; } $expression = $this->parseTypeExpression($typesExpression, null, []); self::assertSame($expectedTypes, $expression->getTypes()); $unionTestNs = '__UnionTest__'; $unionExpression = $this->parseTypeExpression( $unionTestNs.'\\A|'.$typesExpression.'|'.$unionTestNs.'\\Z', null, [] ); self::assertSame( [$unionTestNs.'\\A', ...$expectedTypes, $unionTestNs.'\\Z'], [...$unionExpression->getTypes()] ); } public static function provideGetTypesCases(): iterable { yield ['int']; yield ['Foo5']; yield ['🚀_kůň']; yield ['positive-int']; yield ['?int']; yield ['? int']; yield ['int[]']; yield ['Foo[][]']; yield ['Foo [ ] []']; yield ['int[]|null', ['int[]', 'null']]; yield ['int[]|null|?int|array', ['int[]', 'null', '?int', 'array']]; yield ['null|Foo\Bar|\Baz\Bax|int[]', ['null', 'Foo\Bar', '\Baz\Bax', 'int[]']]; yield ['gen']; yield ['int|gen', ['int', 'gen']]; yield ['\int|\gen<\int, \bool>', ['\int', '\gen<\int, \bool>']]; yield ['gen']; yield ['gen']; yield ['gen']; yield ['gen>']; yield ['gen>']; yield ['gen[][]']; yield ['non-empty-array']; yield ['null|gen>|int|string[]', ['null', 'gen>', 'int', 'string[]']]; yield ['null|gen>|int|array|string[]', ['null', 'gen>', 'int', 'array', 'string[]']]; yield ['this']; yield ['@this']; yield ['$SELF|int', ['$SELF', 'int']]; yield ['array']; yield ['Collection, Foo>']; yield ['int | string', ['int', 'string']]; yield ['Foo::*']; yield ['Foo::A']; yield ['Foo::A|Foo::B', ['Foo::A', 'Foo::B']]; yield ['Foo::A*']; yield ['Foo::*0*_Bar']; yield ['?Foo::*[]']; yield ['array|null', ['array', 'null']]; yield ['null|true|false|1|-1|1.5|-1.5|.5|1.|\'a\'|"b"', ['null', 'true', 'false', '1', '-1', '1.5', '-1.5', '.5', '1.', "'a'", '"b"']]; yield ['int | "a" | A, E>', ['int', '"a"', 'A, E>']]; yield ['class-string']; yield ['A&B', ['A', 'B']]; yield ['A & B', ['A', 'B']]; yield ['array{}']; yield ['object{ }']; yield ['array{1: bool, 2: bool}']; yield ['array{a: int|string, b?: bool}']; yield ['array{\'a\': "a", "b"?: \'b\'}']; yield ['array { a : int | string , b ? : A }']; yield ['array{bool, int}']; yield ['array{bool,}']; yield ['list{int, bool}']; yield ['object{ bool, foo2: int }']; yield ['ArRAY{ 1 }']; yield ['lIst{ 1 }']; yield ['OBJECT { x: 1 }']; yield ['callable']; yield ['callable(string)']; yield ['? callable(string): bool']; yield ['CAllable(string): bool']; yield ['callable(string,): bool']; yield ['callable(array, array): bool']; yield ['array']; yield ['callable(string): callable(int)']; yield ['callable(string) : callable(int) : bool']; yield ['TheCollection|string[]|null', ['TheCollection', 'string[]', 'null']]; yield ['Closure()']; yield ['Closure(string)']; yield ['\\closure(string): void']; yield ['\\Closure']; yield ['\\Closure()']; yield ['\\Closure(string)']; yield ['\\Closure(string, bool)']; yield ['\\Closure(string|int, bool)']; yield ['\\Closure(string):bool']; yield ['\\Closure(string): bool']; yield ['\\Closure(string|int, bool): bool']; yield ['\\Closure(float|int): (bool|int)']; yield ['Closure(int $a)']; yield ['Closure(int $a): bool']; yield ['Closure(int $a, array>): bool']; yield ['Closure_can_be_aliased()']; yield ['Closure_can_be_aliased(): (u|v)']; yield ['array < int , callable ( string ) : bool >']; yield ['(int)']; yield ['(int|\\Exception)']; yield ['($foo is int ? false : true)']; yield ['($foo🚀3 is int ? false : true)']; yield ['\'a\\\'s"\\\\\n\r\t\'|"b\\"s\'\\\\\n\r\t"', ['\'a\\\'s"\\\\\n\r\t\'', '"b\\"s\'\\\\\n\r\t"']]; } public static function provideGetConstTypesCases(): iterable { foreach ([ 'null', 'true', 'FALSE', '123', '+123', '-123', '0b0110101', '0o777', '0x7Fb4', '-0O777', '-0X7Fb4', '123_456', '0b01_01_01', '-0X7_Fb_4', '18_446_744_073_709_551_616', // 64-bit unsigned long + 1, larger than PHP_INT_MAX '123.4', '.123', '123.', '123e4', '123E4', '12.3e4', '+123.5', '-123.', '-123.4', '-.123', '-123.', '-123e-4', '-12.3e-4', '-1_2.3_4e5_6', '123E+80', '8.2023437675747321', // greater precision than 64-bit double '-0.0', '\'\'', '\'foo\'', '\'\\\\\'', '\'\\\'\'', ] as $type) { yield [$type]; } } /** * @dataProvider provideParseInvalidExceptionCases */ public function testParseInvalidException(string $value): void { $this->expectException(\Exception::class); $this->expectExceptionMessage('Unable to parse phpdoc type'); new TypeExpression($value, null, []); } public static function provideParseInvalidExceptionCases(): iterable { yield ['']; yield ['0_class_cannot_start_with_number']; yield ['$0_variable_cannot_start_with_number']; yield ['class cannot contain space']; yield ['\\\\class_with_double_backslash']; yield ['class\\\\with_double_backslash']; yield ['class_with_end_backslash\\']; yield ['class/with_slash']; yield ['class--with_double_dash']; yield ['class.with_dot']; yield ['class,with_comma']; yield ['class@with_at_sign']; yield ['class:with_colon']; yield ['class#with_hash']; yield ['class//with_double_slash']; yield ['class$with_dollar']; yield ['class:with_colon']; yield ['class;with_semicolon']; yield ['class=with_equal_sign']; yield ['class+with_plus']; yield ['class?with_question_mark']; yield ['class*with_star']; yield ['class%with_percent']; yield ['(unclosed_parenthesis']; yield [')unclosed_parenthesis']; yield ['unclosed_parenthesis(']; yield ['((unclosed_parenthesis)']; yield ['array<']; yield ['array<<']; yield ['array>']; yield ['array<<>']; yield ['array<>>']; yield ['array{']; yield ['array{ $this: 5 }']; yield ['g<,>']; yield ['g<,no_leading_comma>']; yield ['10__000']; yield ['[ array_syntax_is_invalid ]']; yield ['\' unclosed string']; yield ['\' unclosed string \\\'']; yield 'generic with no arguments' => ['f<>']; } public function testHugeType(): void { $nFlat = 2_000; $types = []; for ($i = 0; $i < $nFlat; ++$i) { $types[] = '\X\Foo'.$i; } $str = implode('|', $types); $expression = new TypeExpression($str, null, []); self::assertSame($types, $expression->getTypes()); $nRecursive = 100; for ($i = 0; $i < $nRecursive; ++$i) { $str = 'array'.(1 === $i % 2 ? '{' : '<').$str.(1 === $i % 2 ? '}' : '>'); } $typeLeft = '\Closure(A|B): void'; $typeRight = '\Closure('.$typeLeft.'): void'; $expression = new TypeExpression($typeLeft.'|('.$str.')|'.$typeRight, null, []); self::assertSame([$typeLeft, '('.$str.')', $typeRight], $expression->getTypes()); } /** * @dataProvider provideGetTypesGlueCases */ public function testGetTypesGlue(string $expectedTypesGlue, string $typesExpression): void { $expression = new TypeExpression($typesExpression, null, []); self::assertSame($expectedTypesGlue, $expression->getTypesGlue()); } public static function provideGetTypesGlueCases(): iterable { yield ['|', 'string']; // for backward behaviour yield ['|', 'bool|string']; yield ['&', 'Foo&Bar']; } /** * @param NamespaceUseAnalysis[] $namespaceUses * * @dataProvider provideGetCommonTypeCases */ public function testGetCommonType(string $typesExpression, ?string $expectedCommonType, NamespaceAnalysis $namespace = null, array $namespaceUses = []): void { $expression = new TypeExpression($typesExpression, $namespace, $namespaceUses); self::assertSame($expectedCommonType, $expression->getCommonType()); } public static function provideGetCommonTypeCases(): iterable { $globalNamespace = new NamespaceAnalysis('', '', 0, 999, 0, 999); $appNamespace = new NamespaceAnalysis('App', 'App', 0, 999, 0, 999); $useTraversable = new NamespaceUseAnalysis('Traversable', 'Traversable', false, 0, 0, NamespaceUseAnalysis::TYPE_CLASS); $useObjectAsTraversable = new NamespaceUseAnalysis('Foo', 'Traversable', false, 0, 0, NamespaceUseAnalysis::TYPE_CLASS); yield ['true', 'bool']; yield ['false', 'bool']; yield ['bool', 'bool']; yield ['int', 'int']; yield ['float', 'float']; yield ['string', 'string']; yield ['array', 'array']; yield ['object', 'object']; yield ['self', 'self']; yield ['static', 'static']; yield ['bool[]', 'array']; yield ['int[]', 'array']; yield ['float[]', 'array']; yield ['string[]', 'array']; yield ['array[]', 'array']; yield ['bool[][]', 'array']; yield ['int[][]', 'array']; yield ['float[][]', 'array']; yield ['string[][]', 'array']; yield ['array[][]', 'array']; yield ['bool [ ]', 'array']; yield ['bool [ ][ ]', 'array']; yield ['array|iterable', 'iterable']; yield ['iterable|array', 'iterable']; yield ['array|Traversable', 'iterable']; yield ['array|\Traversable', 'iterable']; yield ['array|Traversable', 'iterable', $globalNamespace]; yield ['iterable|Traversable', 'iterable']; yield ['array', 'array']; yield ['array', 'array']; yield ['array < string >', 'array']; yield ['list', 'array']; yield ['iterable', 'iterable']; yield ['iterable', 'iterable']; yield ['\Traversable', '\Traversable']; yield ['Traversable', 'Traversable']; yield ['Collection', 'Collection']; yield ['Collection', 'Collection']; yield ['array{string}', 'array']; yield ['array { 1: string, \Closure(): void }', 'array']; yield ['Closure(): void', 'Closure']; yield ['array|iterable', 'iterable']; yield ['int[]|string[]', 'array']; yield ['int|null', 'int']; yield ['null|int', 'int']; yield ['?int', 'int']; yield ['?array', 'array']; yield ['?list', 'array']; yield ['void', 'void']; yield ['never', 'never']; yield ['array|Traversable', 'iterable', null, [$useTraversable]]; yield ['array|Traversable', 'iterable', $globalNamespace, [$useTraversable]]; yield ['array|Traversable', 'iterable', $appNamespace, [$useTraversable]]; yield ['self|static', 'self']; yield ['array|Traversable', null, null, [$useObjectAsTraversable]]; yield ['array|Traversable', null, $globalNamespace, [$useObjectAsTraversable]]; yield ['array|Traversable', null, $appNamespace, [$useObjectAsTraversable]]; yield ['bool|int', null]; yield ['string|bool', null]; yield ['array|Collection', null]; } /** * @dataProvider provideAllowsNullCases */ public function testAllowsNull(string $typesExpression, bool $expectNullAllowed): void { $expression = new TypeExpression($typesExpression, null, []); self::assertSame($expectNullAllowed, $expression->allowsNull()); } public static function provideAllowsNullCases(): iterable { yield ['null', true]; yield ['mixed', true]; yield ['null|mixed', true]; yield ['int|bool|null', true]; yield ['int|bool|mixed', true]; yield ['int', false]; yield ['bool', false]; yield ['string', false]; yield ['?int', true]; yield ['?\Closure(): void', true]; } /** * @dataProvider provideSortTypesCases */ public function testSortTypes(string $typesExpression, string $expectResult): void { $sortCaseFx = static fn (TypeExpression $a, TypeExpression $b): int => strcasecmp($a->toString(), $b->toString()); $sortCrc32Fx = static fn (TypeExpression $a, TypeExpression $b): int => crc32($a->toString()) <=> crc32($b->toString()); $expression = $this->parseTypeExpression($typesExpression, null, []); $expression->sortTypes($sortCaseFx); self::assertSame($expectResult, $expression->toString()); $expression->sortTypes($sortCrc32Fx); $expression->sortTypes($sortCaseFx); self::assertSame($expectResult, $expression->toString()); } public static function provideSortTypesCases(): iterable { yield 'not a union type' => [ 'int', 'int', ]; yield 'simple' => [ 'int|bool', 'bool|int', ]; yield 'multiple union' => [ 'C___|D____|B__|A', 'A|B__|C___|D____', ]; yield 'multiple intersect' => [ 'C___&D____&B__&A', 'A&B__&C___&D____', ]; yield 'simple in generic' => [ 'array', 'array', ]; yield 'generic with multiple types' => [ 'array', 'array', ]; yield 'generic with trailing comma' => [ 'array', 'array', ]; yield 'simple in array shape with int key' => [ 'array{0: int|bool}', 'array{0: bool|int}', ]; yield 'simple in array shape with string key' => [ 'array{"foo": int|bool}', 'array{"foo": bool|int}', ]; yield 'simple in array shape with multiple keys' => [ 'array{0: int|bool, "foo": int|bool}', 'array{0: bool|int, "foo": bool|int}', ]; yield 'simple in array shape with implicit key' => [ 'array{int|bool}', 'array{bool|int}', ]; yield 'simple in array shape with trailing comma' => [ 'array{int|bool,}', 'array{bool|int,}', ]; yield 'simple in array shape with multiple types with trailing comma' => [ 'array{int|bool, Foo|Bar, }', 'array{bool|int, Bar|Foo, }', ]; yield 'simple in array shape' => [ 'list{int, Foo|Bar}', 'list{int, Bar|Foo}', ]; yield 'array shape with multiple colons - array shape' => [ 'array{array{x:int|bool}, a:array{x:int|bool}}', 'array{array{x:bool|int}, a:array{x:bool|int}}', ]; yield 'array shape with multiple colons - callable' => [ 'array{array{x:int|bool}, int|bool, callable(): void}', 'array{array{x:bool|int}, bool|int, callable(): void}', ]; yield 'simple in callable argument' => [ 'callable(int|bool)', 'callable(bool|int)', ]; yield 'callable with multiple arguments' => [ 'callable(int|bool, null|array)', 'callable(bool|int, array|null)', ]; yield 'simple in callable return type' => [ 'callable(): (string|float)', 'callable(): (float|string)', ]; yield 'callable with union return type and within union itself' => [ 'callable(): (string|float)|bool', 'bool|callable(): (float|string)', ]; yield 'callable with multiple named arguments' => [ 'callable(int|bool $b, null|array $a)', 'callable(bool|int $b, array|null $a)', ]; yield 'callable with complex arguments' => [ 'callable(B|A&, D|Closure(): void..., array{}$foo=, $this $foo=): array{}', 'callable(A|B&, Closure(): void|D..., array{}$foo=, $this $foo=): array{}', ]; yield 'callable with trailing comma' => [ 'Closure( Y|X , ): B|A', 'A|Closure( X|Y , ): B', ]; yield 'simple in Closure argument' => [ 'Closure(int|bool)', 'Closure(bool|int)', ]; yield 'Closure with multiple arguments' => [ 'Closure(int|bool, null|array)', 'Closure(bool|int, array|null)', ]; yield 'simple in Closure argument with trailing comma' => [ 'Closure(int|bool,)', 'Closure(bool|int,)', ]; yield 'simple in Closure argument multiple arguments with trailing comma' => [ 'Closure(int|bool, null|array,)', 'Closure(bool|int, array|null,)', ]; yield 'simple in Closure return type' => [ 'Closure(): (string|float)', 'Closure(): (float|string)', ]; yield 'Closure with union return type and within union itself' => [ 'Closure(): (string|float)|bool', 'bool|Closure(): (float|string)', ]; yield 'with multiple nesting levels' => [ 'array{0: Foo|Bar): (Foo|Bar)>}', 'array{0: Bar|float|string): (Bar|Foo)>|Foo}', ]; yield 'with multiple nesting levels and callable within union' => [ 'array{0: Foo|Bar): (Foo|Bar)|Baz>}', 'array{0: Bar|float|string): (Bar|Foo)>|Foo}', ]; yield 'complex type with Closure with $this' => [ 'array|false', 'array|false', ]; yield 'nullable generic' => [ '?array', '?array', ]; yield 'nullable callable' => [ '?callable(Foo|Bar): (Foo|Bar)', '?callable(Bar|Foo): (Bar|Foo)', ]; // This union type makes no sense in general (it should be `Bar|callable|null`) // but let's ensure nullable types are also sorted. yield 'nullable callable with union return type and within union itself' => [ '?callable(Foo|Bar): (Foo|Bar)|?Bar', '?Bar|?callable(Bar|Foo): (Bar|Foo)', ]; yield 'nullable array shape' => [ '?array{0: Foo|Bar}', '?array{0: Bar|Foo}', ]; yield 'simple types alternation' => [ 'array', 'array', ]; yield 'nesty stuff' => [ 'array>>', 'array|Level2>&Level11>', ]; yield 'parenthesized' => [ '(Foo|Bar)', '(Bar|Foo)', ]; yield 'parenthesized intersect' => [ '(Foo&Bar)', '(Bar&Foo)', ]; yield 'parenthesized in closure return type' => [ 'Closure(Y|X): (string|float)', 'Closure(X|Y): (float|string)', ]; yield 'conditional with variable' => [ '($x is (CFoo|(CBaz&CBar)) ? (TFoo|(TBaz&TBar)) : (FFoo|(FBaz&FBar)))', '($x is ((CBar&CBaz)|CFoo) ? ((TBar&TBaz)|TFoo) : ((FBar&FBaz)|FFoo))', ]; yield 'conditional with type' => [ '((Foo|Bar) is x ? y : z)', '((Bar|Foo) is x ? y : z)', ]; yield 'conditional in conditional' => [ '((Foo|Bar) is x ? ($x is (CFoo|CBar) ? (TFoo|TBar) : (FFoo|FBar)) : z)', '((Bar|Foo) is x ? ($x is (CBar|CFoo) ? (TBar|TFoo) : (FBar|FFoo)) : z)', ]; yield 'large numbers' => [ '18_446_744_073_709_551_616|-8.2023437675747321e-18_446_744_073_709_551_616', '-8.2023437675747321e-18_446_744_073_709_551_616|18_446_744_073_709_551_616', ]; } /** * @return list */ private function checkInnerTypeExpressionsStartIndex(TypeExpression $typeExpression): array { $innerTypeExpressions = \Closure::bind(static fn () => $typeExpression->innerTypeExpressions, null, TypeExpression::class)(); $res = []; foreach ($innerTypeExpressions as ['start_index' => $innerStartIndex, 'expression' => $innerExpression]) { $innerExpressionStr = $innerExpression->toString(); self::assertSame( $innerExpressionStr, substr($typeExpression->toString(), $innerStartIndex, \strlen($innerExpressionStr)) ); $res[] = [$innerStartIndex, $innerExpressionStr]; $res[] = $this->checkInnerTypeExpressionsStartIndex($innerExpression); } return $res; } /** * Should be removed once https://github.com/php/php-src/pull/11396 is merged. */ private function clearPcreRegexCache(): void { // there is no explicit php function to clear PCRE regex cache, but based // on https://www.php.net/manual/en/intro.pcre.php there are 4096 cache slots // pruned in FIFO fashion, so to clear the cache, replace all existing // cache slots with dummy regexes for ($i = 0; $i < 4096; ++$i) { preg_match('/^'.$i.'/', ''); } } /** * Parse type expression with and without PCRE JIT. * * @param NamespaceUseAnalysis[] $namespaceUses */ private function parseTypeExpression(string $value, ?NamespaceAnalysis $namespace, array $namespaceUses): TypeExpression { $pcreJitBackup = \ini_get('pcre.jit'); $expression = null; $innerExpressionsDataWithoutJit = null; try { foreach ([false, true] as $pcreJit) { ini_set('pcre.jit', $pcreJit ? '1' : '0'); $this->clearPcreRegexCache(); $expression = new TypeExpression($value, null, []); $innerExpressionsData = $this->checkInnerTypeExpressionsStartIndex($expression); if (false === $pcreJit) { $innerExpressionsDataWithoutJit = $innerExpressionsData; } else { self::assertSame($innerExpressionsDataWithoutJit, $innerExpressionsData); } } } finally { ini_set('pcre.jit', $pcreJitBackup); $this->clearPcreRegexCache(); } return $expression; } }