123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855 |
- <?php
- declare(strict_types=1);
- /*
- * This file is part of PHP CS Fixer.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- * Dariusz Rumiński <dariusz.ruminski@gmail.com>
- *
- * This source file is subject to the MIT license that is bundled
- * with this source code in the file LICENSE.
- */
- namespace PhpCsFixer\Tests\Tokenizer\Analyzer;
- use PhpCsFixer\Tests\TestCase;
- use PhpCsFixer\Tokenizer\Analyzer\Analysis\ArgumentAnalysis;
- use PhpCsFixer\Tokenizer\Analyzer\Analysis\TypeAnalysis;
- use PhpCsFixer\Tokenizer\Analyzer\FunctionsAnalyzer;
- use PhpCsFixer\Tokenizer\Tokens;
- /**
- * @author VeeWee <toonverwerft@gmail.com>
- *
- * @internal
- *
- * @covers \PhpCsFixer\Tokenizer\Analyzer\FunctionsAnalyzer
- */
- final class FunctionsAnalyzerTest extends TestCase
- {
- /**
- * @param list<int> $indices
- *
- * @dataProvider provideIsGlobalFunctionCallCases
- */
- public function testIsGlobalFunctionCall(string $code, array $indices): void
- {
- self::assertIsGlobalFunctionCall($indices, $code);
- }
- /**
- * @return iterable<array{string, list<int>}>
- */
- public static function provideIsGlobalFunctionCallCases(): iterable
- {
- yield [
- '<?php CONSTANT;',
- [],
- ];
- yield [
- '<?php foo();',
- [1],
- ];
- yield [
- '<?php foo("bar");',
- [1],
- ];
- yield [
- '<?php \foo("bar");',
- [2],
- ];
- yield [
- '<?php foo\bar("baz");',
- [],
- ];
- yield [
- '<?php foo::bar("baz");',
- [],
- ];
- yield [
- '<?php $foo->bar("baz");',
- [],
- ];
- yield [
- '<?php new bar("baz");',
- [],
- ];
- yield [
- '<?php function foo() {}',
- [],
- ];
- yield 'function with ref. return' => [
- '<?php function & foo() {}',
- [],
- ];
- yield [
- '<?php namespace\foo("bar");',
- [],
- ];
- yield [
- '<?php
- namespace A {
- use function A;
- }
- namespace B {
- use function D;
- A();
- }
- ',
- [30],
- ];
- yield [
- '<?php
- function A(){}
- A();
- ',
- [10],
- ];
- yield [
- '<?php
- function A(){}
- a();
- ',
- [10],
- ];
- yield [
- '<?php
- namespace {
- function A(){}
- A();
- }
- ',
- [14],
- ];
- yield [
- '<?php
- namespace Z {
- function A(){}
- A();
- }
- ',
- [],
- ];
- yield [
- '<?php
- namespace Z;
- function A(){}
- A();
- ',
- [],
- ];
- yield 'function signature ref. return, calls itself' => [
- '<?php
- function & A(){}
- A();
- ',
- [12],
- ];
- yield [
- '<?php
- class Foo
- {
- public function A(){}
- }
- A();
- ',
- [20],
- ];
- yield [
- '<?php
- namespace A {
- function A(){}
- }
- namespace B {
- A();
- }
- ',
- [24],
- ];
- yield [
- '<?php
- use function X\a;
- A();
- ',
- [],
- ];
- yield [
- '<?php
- use A;
- A();
- ',
- [7],
- ];
- yield [
- '<?php
- use const A;
- A();
- ',
- [9],
- ];
- yield [
- '<?php
- use function A;
- str_repeat($a, $b);
- ',
- [9],
- ];
- yield [
- '<?php
- namespace {
- function A(){}
- A();
- $b = function(){};
- }
- ',
- [14],
- ];
- yield [
- '<?php implode($a);implode($a);implode($a);implode($a);implode($a);implode($a);',
- [1, 6, 11, 16, 21, 26],
- ];
- yield [
- '<?php
- $z = new class(
- new class(){ private function A(){} }
- ){
- public function A() {}
- };
- A();
- ',
- [46],
- ];
- yield [
- '<?php $foo = fn() => false;',
- [],
- ];
- yield [
- '<?php foo("bar"); class A { function Foo(){ foo(); } }',
- [1, 20],
- ];
- }
- /**
- * @param list<int> $indices
- *
- * @dataProvider provideIsGlobalFunctionCallPre80Cases
- *
- * @requires PHP <8.0
- */
- public function testIsGlobalFunctionCallPre80(string $code, array $indices): void
- {
- self::assertIsGlobalFunctionCall($indices, $code);
- }
- /**
- * @return iterable<array{string, list<int>}>
- */
- public static function provideIsGlobalFunctionCallPre80Cases(): iterable
- {
- yield [
- '<?php
- use function \ str_repeat;
- str_repeat($a, $b);
- ',
- [11],
- ];
- }
- /**
- * @param list<int> $indices
- *
- * @dataProvider provideIsGlobalFunctionCallPhp80Cases
- *
- * @requires PHP 8.0
- */
- public function testIsGlobalFunctionCallPhp80(string $code, array $indices): void
- {
- self::assertIsGlobalFunctionCall($indices, $code);
- }
- public static function provideIsGlobalFunctionCallPhp80Cases(): iterable
- {
- yield [
- '<?php $a = new (foo());',
- [8],
- ];
- yield [
- '<?php $b = $foo instanceof (foo());',
- [10],
- ];
- yield [
- '<?php
- #[\Attribute(\Attribute::TARGET_CLASS)]
- class Foo {}
- ',
- [],
- ];
- yield [
- '<?php $x?->count();',
- [],
- ];
- yield [
- '<?php
- #[Foo(), Bar(), Baz()]
- class Foo {}
- ',
- [],
- ];
- }
- /**
- * @param list<int> $indices
- *
- * @dataProvider provideIsGlobalFunctionCallPhp81Cases
- *
- * @requires PHP 8.1
- */
- public function testIsGlobalFunctionCallPhp81(array $indices, string $code): void
- {
- self::assertIsGlobalFunctionCall($indices, $code);
- }
- public static function provideIsGlobalFunctionCallPhp81Cases(): iterable
- {
- yield 'first class callable cases' => [
- [],
- '<?php
- strlen(...);
- \strlen(...);
- $closure(...);
- $invokableObject(...);
- $obj->method(...);
- $obj->$methodStr(...);
- ($obj->property)(...);
- Foo::method(...);
- $classStr::$methodStr(...);
- self::{$complex . $expression}(...);
- \'strlen\'(...);
- [$obj, \'method\'](...);
- [Foo::class, \'method\'](...);
- $c = new class{};
- $b = new class(){};
- $a = new #[foo]
- class(){};
- ',
- ];
- yield [
- [1, 20],
- '<?php foo("bar"); enum A { function Foo(){ foo(); } }',
- ];
- }
- /**
- * @param array<string, ArgumentAnalysis> $expected
- *
- * @dataProvider provideFunctionArgumentInfoCases
- */
- public function testFunctionArgumentInfo(string $code, int $methodIndex, array $expected): void
- {
- $tokens = Tokens::fromCode($code);
- $analyzer = new FunctionsAnalyzer();
- self::assertSame(serialize($expected), serialize($analyzer->getFunctionArguments($tokens, $methodIndex)));
- }
- /**
- * @return iterable<array{string, int, array<string, ArgumentAnalysis>}>
- */
- public static function provideFunctionArgumentInfoCases(): iterable
- {
- yield ['<?php function(){};', 1, []];
- yield ['<?php function($a){};', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 3,
- null,
- null
- ),
- ]];
- yield ['<?php function($a, $b){};', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 3,
- null,
- null
- ),
- '$b' => new ArgumentAnalysis(
- '$b',
- 6,
- null,
- null
- ),
- ]];
- yield ['<?php function($a, $b = array(1,2), $c = 3){};', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 3,
- null,
- null
- ),
- '$b' => new ArgumentAnalysis(
- '$b',
- 6,
- 'array(1,2)',
- null
- ),
- '$c' => new ArgumentAnalysis(
- '$c',
- 18,
- '3',
- null
- ),
- ]];
- yield ['<?php function(array $a = array()){};', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 5,
- 'array()',
- new TypeAnalysis(
- 'array',
- 3,
- 3
- )
- ),
- ]];
- yield ['<?php function(array ... $a){};', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 7,
- null,
- new TypeAnalysis(
- 'array',
- 3,
- 3
- )
- ),
- ]];
- yield ['<?php function(\Foo\Bar $a){};', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 8,
- null,
- new TypeAnalysis(
- '\Foo\Bar',
- 3,
- 6
- )
- ),
- ]];
- yield ['<?php fn() => null;', 1, []];
- yield ['<?php fn($a) => null;', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 3,
- null,
- null
- ),
- ]];
- yield ['<?php fn($a, $b) => null;', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 3,
- null,
- null
- ),
- '$b' => new ArgumentAnalysis(
- '$b',
- 6,
- null,
- null
- ),
- ]];
- yield ['<?php fn($a, $b = array(1,2), $c = 3) => null;', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 3,
- null,
- null
- ),
- '$b' => new ArgumentAnalysis(
- '$b',
- 6,
- 'array(1,2)',
- null
- ),
- '$c' => new ArgumentAnalysis(
- '$c',
- 18,
- '3',
- null
- ),
- ]];
- yield ['<?php fn(array $a = array()) => null;', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 5,
- 'array()',
- new TypeAnalysis(
- 'array',
- 3,
- 3
- )
- ),
- ]];
- yield ['<?php fn(array ... $a) => null;', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 7,
- null,
- new TypeAnalysis(
- 'array',
- 3,
- 3
- )
- ),
- ]];
- yield ['<?php fn(\Foo\Bar $a) => null;', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 8,
- null,
- new TypeAnalysis(
- '\Foo\Bar',
- 3,
- 6
- )
- ),
- ]];
- }
- /**
- * @param array<string, ArgumentAnalysis> $expected
- *
- * @dataProvider provideFunctionArgumentInfoPre80Cases
- *
- * @requires PHP <8.0
- */
- public function testFunctionArgumentInfoPre80(string $code, int $methodIndex, array $expected): void
- {
- $this->testFunctionArgumentInfo($code, $methodIndex, $expected);
- }
- /**
- * @return iterable<array{string, int, array<string, ArgumentAnalysis>}>
- */
- public static function provideFunctionArgumentInfoPre80Cases(): iterable
- {
- yield ['<?php fn(\Foo/** TODO: change to something else */\Bar $a) => null;', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 9,
- null,
- new TypeAnalysis(
- '\Foo\Bar',
- 3,
- 7
- )
- ),
- ]];
- yield ['<?php function(\Foo/** TODO: change to something else */\Bar $a){};', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 9,
- null,
- new TypeAnalysis(
- '\Foo\Bar',
- 3,
- 7
- )
- ),
- ]];
- }
- /**
- * @dataProvider provideFunctionReturnTypeInfoCases
- */
- public function testFunctionReturnTypeInfo(string $code, int $methodIndex, ?TypeAnalysis $expected): void
- {
- $tokens = Tokens::fromCode($code);
- $analyzer = new FunctionsAnalyzer();
- $actual = $analyzer->getFunctionReturnType($tokens, $methodIndex);
- self::assertSame(serialize($expected), serialize($actual));
- }
- /**
- * @return iterable<array{string, int, null|TypeAnalysis}>
- */
- public static function provideFunctionReturnTypeInfoCases(): iterable
- {
- yield ['<?php function(){};', 1, null];
- yield ['<?php function($a): array {};', 1, new TypeAnalysis('array', 7, 7)];
- yield ['<?php function($a): \Foo\Bar {};', 1, new TypeAnalysis('\Foo\Bar', 7, 10)];
- yield ['<?php function($a): /* not sure if really an array */array {};', 1, new TypeAnalysis('array', 8, 8)];
- yield ['<?php fn() => null;', 1, null];
- yield ['<?php fn(array $a) => null;', 1, null];
- yield ['<?php fn($a): array => null;', 1, new TypeAnalysis('array', 7, 7)];
- yield ['<?php fn($a): \Foo\Bar => null;', 1, new TypeAnalysis('\Foo\Bar', 7, 10)];
- yield ['<?php fn($a): /* not sure if really an array */array => null;', 1, new TypeAnalysis('array', 8, 8)];
- }
- /**
- * @dataProvider provideFunctionReturnTypeInfoPre80Cases
- *
- * @requires PHP <8.0
- */
- public function testFunctionReturnTypeInfoPre80(string $code, int $methodIndex, ?TypeAnalysis $expected): void
- {
- $this->testFunctionReturnTypeInfo($code, $methodIndex, $expected);
- }
- /**
- * @return iterable<array{string, int, null|TypeAnalysis}>
- */
- public static function provideFunctionReturnTypeInfoPre80Cases(): iterable
- {
- yield ['<?php function($a): \Foo/** TODO: change to something else */\Bar {};', 1, new TypeAnalysis('\Foo\Bar', 7, 11)];
- yield ['<?php fn($a): \Foo/** TODO: change to something else */\Bar => null;', 1, new TypeAnalysis('\Foo\Bar', 7, 11)];
- }
- public function testIsTheSameClassCallInvalidIndex(): void
- {
- $tokens = Tokens::fromCode('<?php 1;2;');
- $analyzer = new FunctionsAnalyzer();
- $this->expectException(\InvalidArgumentException::class);
- $this->expectExceptionMessage('Token index 666 does not exist.');
- $analyzer->isTheSameClassCall($tokens, 666);
- }
- /**
- * @dataProvider provideIsTheSameClassCallCases
- *
- * @param list<int> $sameClassCallIndices
- */
- public function testIsTheSameClassCall(string $code, array $sameClassCallIndices): void
- {
- $tokens = Tokens::fromCode($code);
- $analyzer = new FunctionsAnalyzer();
- for ($index = $tokens->count() - 1; $index >= 0; --$index) {
- self::assertSame(
- \in_array($index, $sameClassCallIndices, true),
- $analyzer->isTheSameClassCall($tokens, $index),
- \sprintf('Index %d failed check.', $index)
- );
- }
- }
- /**
- * @return iterable<array{string, list<int>}>
- */
- public static function provideIsTheSameClassCallCases(): iterable
- {
- $template = '<?php
- class Foo {
- public function methodOne() {
- $x = %sotherMethod(1, 2, 3);
- }
- }
- ';
- yield [
- \sprintf($template, '$this->'),
- [24],
- ];
- yield [
- \sprintf($template, 'self::'),
- [24],
- ];
- yield [
- \sprintf($template, 'static::'),
- [24],
- ];
- yield [
- \sprintf($template, '$THIS->'),
- [24],
- ];
- yield [
- \sprintf($template, '$notThis->'),
- [],
- ];
- yield [
- \sprintf($template, 'Bar::'),
- [],
- ];
- yield [
- \sprintf($template, '$this::'),
- [24],
- ];
- yield [
- <<<'PHP'
- <?php
- class Foo {
- private $bar;
- public function bar() {
- return $this->bar;
- }
- }
- PHP,
- [],
- ];
- }
- /**
- * @dataProvider provideIsTheSameClassCall80Cases
- *
- * @param list<int> $sameClassCallIndices
- *
- * @requires PHP 8.0
- */
- public function testIsTheSameClassCall80(string $code, array $sameClassCallIndices): void
- {
- $this->testIsTheSameClassCall($code, $sameClassCallIndices);
- }
- /**
- * @return iterable<array{string, list<int>}>
- */
- public static function provideIsTheSameClassCall80Cases(): iterable
- {
- yield [
- '<?php
- class Foo {
- public function methodOne() {
- $x = $this?->otherMethod(1, 2, 3);
- }
- }
- ',
- [24],
- ];
- }
- /**
- * @param array<string, ArgumentAnalysis> $expected
- *
- * @dataProvider provideFunctionArgumentInfoPhp80Cases
- *
- * @requires PHP 8.0
- */
- public function testFunctionArgumentInfoPhp80(string $code, int $methodIndex, array $expected): void
- {
- $this->testFunctionArgumentInfo($code, $methodIndex, $expected);
- }
- public static function provideFunctionArgumentInfoPhp80Cases(): iterable
- {
- yield ['<?php function($aa,){};', 1, [
- '$aa' => new ArgumentAnalysis(
- '$aa',
- 3,
- null,
- null
- ),
- ]];
- yield ['<?php fn($a, $bc ,) => null;', 1, [
- '$a' => new ArgumentAnalysis(
- '$a',
- 3,
- null,
- null
- ),
- '$bc' => new ArgumentAnalysis(
- '$bc',
- 6,
- null,
- null
- ),
- ]];
- }
- /**
- * @param list<int> $expectedIndices
- */
- private static function assertIsGlobalFunctionCall(array $expectedIndices, string $code): void
- {
- $tokens = Tokens::fromCode($code);
- $analyzer = new FunctionsAnalyzer();
- $actualIndices = [];
- foreach ($tokens as $index => $token) {
- if ($analyzer->isGlobalFunctionCall($tokens, $index)) {
- $actualIndices[] = $index;
- }
- }
- self::assertSame(
- $expectedIndices,
- $actualIndices,
- \sprintf(
- 'Global function calls found at positions: [%s], expected at [%s].',
- implode(', ', $actualIndices),
- implode(', ', $expectedIndices)
- )
- );
- }
- }
|