From 0fe2ec56c897612422d8b75b10907cc02dad8046 Mon Sep 17 00:00:00 2001 From: ErickSkrauch Date: Fri, 19 Jan 2024 02:57:40 +0100 Subject: [PATCH] Resolve TODOs, cover edge cases, add more tests --- .php-cs-fixer.dist.php | 4 +- README.md | 2 + src/Analyzer/ClassElementsAnalyzer.php | 132 ++++++++++++++++++ src/Analyzer/ClassNameAnalyzer.php | 11 +- src/ClassNotation/OrderedOverridesFixer.php | 112 ++++++--------- tests/Analyzer/ClassNameAnalyzerTest.php | 82 +++++++++++ .../OrderedOverridesFixerTest.php | 49 +++++++ .../_data/InterfaceAReverseOrder.php | 12 ++ 8 files changed, 323 insertions(+), 81 deletions(-) create mode 100644 src/Analyzer/ClassElementsAnalyzer.php create mode 100644 tests/Analyzer/ClassNameAnalyzerTest.php create mode 100644 tests/ClassNotation/_data/InterfaceAReverseOrder.php diff --git a/.php-cs-fixer.dist.php b/.php-cs-fixer.dist.php index d26d7a1..bc18da3 100644 --- a/.php-cs-fixer.dist.php +++ b/.php-cs-fixer.dist.php @@ -2,7 +2,9 @@ declare(strict_types=1); $finder = PhpCsFixer\Finder::create() - ->in(__DIR__); + ->in(__DIR__) + ->exclude('tests/ClassNotation/_data') +; return Ely\CS\Config::create([ // Disable "parameters" and "match" to keep compatibility with PHP 7.4 diff --git a/README.md b/README.md index d0f022a..7c67288 100644 --- a/README.md +++ b/README.md @@ -168,6 +168,8 @@ Ensures that multiline if statement body curly brace placed on the right line. Overridden and implemented methods must be sorted in the same order as they are defined in parent classes. +**Warning**: this fixer is implemented against the PHP-CS-Fixer principle and relies on runtime, classes autoloading and reflection. If dependencies are missing or the autoloader isn't configured correctly, the fixer will not be able to discover the order of methods in parents. + ```diff --- Original +++ New diff --git a/src/Analyzer/ClassElementsAnalyzer.php b/src/Analyzer/ClassElementsAnalyzer.php new file mode 100644 index 0000000..b3f9c43 --- /dev/null +++ b/src/Analyzer/ClassElementsAnalyzer.php @@ -0,0 +1,132 @@ + + */ + public function getClassElements(Tokens $tokens, int $classOpenBraceIndex): array { + static $elementTokenKinds = [CT::T_USE_TRAIT, T_CASE, T_CONST, T_VARIABLE, T_FUNCTION]; + + $startIndex = $classOpenBraceIndex + 1; + $elements = []; + + while (true) { + $element = [ + 'start' => $startIndex, + 'visibility' => 'public', + 'abstract' => false, + 'static' => false, + 'readonly' => false, + ]; + + for ($i = $startIndex; ; ++$i) { + $token = $tokens[$i]; + + // class end + if ($token->equals('}')) { + return $elements; // @phpstan-ignore return.type + } + + if ($token->isGivenKind(T_ABSTRACT)) { + $element['abstract'] = true; + + continue; + } + + if ($token->isGivenKind(T_STATIC)) { + $element['static'] = true; + + continue; + } + + if (defined('T_READONLY') && $token->isGivenKind(T_READONLY)) { + $element['readonly'] = true; + } + + if ($token->isGivenKind([T_PROTECTED, T_PRIVATE])) { + $element['visibility'] = strtolower($token->getContent()); + + continue; + } + + if (!$token->isGivenKind($elementTokenKinds)) { + continue; + } + + $element['type'] = $this->detectElementType($tokens, $i); + if ($element['type'] === 'property') { + $element['name'] = $tokens[$i]->getContent(); + } elseif (in_array($element['type'], ['use_trait', 'case', 'constant', 'method', 'magic', 'construct', 'destruct'], true)) { + $element['name'] = $tokens[$tokens->getNextMeaningfulToken($i)]->getContent(); + } + + $element['end'] = $this->findElementEnd($tokens, $i); + + break; + } + + $elements[] = $element; + $startIndex = $element['end'] + 1; // @phpstan-ignore offsetAccess.notFound + } + } + + /** + * @phpstan-return AnalyzedClassElementType + */ + private function detectElementType(Tokens $tokens, int $index): string { + $token = $tokens[$index]; + if ($token->isGivenKind(CT::T_USE_TRAIT)) { + return 'use_trait'; + } + + if ($token->isGivenKind(T_CASE)) { + return 'case'; + } + + if ($token->isGivenKind(T_CONST)) { + return 'constant'; + } + + if ($token->isGivenKind(T_VARIABLE)) { + return 'property'; + } + + return 'method'; + } + + private function findElementEnd(Tokens $tokens, int $index): int { + $endIndex = $tokens->getNextTokenOfKind($index, ['{', ';']); + if ($tokens[$endIndex]->equals('{')) { + $endIndex = $tokens->findBlockEnd(Tokens::BLOCK_TYPE_CURLY_BRACE, $endIndex); + } + + for (++$endIndex; $tokens[$endIndex]->isWhitespace(" \t") || $tokens[$endIndex]->isComment(); ++$endIndex); + + --$endIndex; + + return $tokens[$endIndex]->isWhitespace() ? $endIndex - 1 : $endIndex; + } + +} diff --git a/src/Analyzer/ClassNameAnalyzer.php b/src/Analyzer/ClassNameAnalyzer.php index 1434e14..df4d48b 100644 --- a/src/Analyzer/ClassNameAnalyzer.php +++ b/src/Analyzer/ClassNameAnalyzer.php @@ -8,8 +8,6 @@ use PhpCsFixer\Tokenizer\Analyzer\NamespaceUsesAnalyzer; use PhpCsFixer\Tokenizer\Tokens; -// TODO: better naming -// TODO: cover with tests final class ClassNameAnalyzer { private NamespacesAnalyzer $namespacesAnalyzer; @@ -58,13 +56,8 @@ private function readClassName(Tokens $tokens, int $classNameStart): string { $className = ''; $index = $classNameStart; do { - $token = $tokens[$index]; - if ($token->isWhitespace()) { - continue; - } - - $className .= $token->getContent(); - } while ($tokens[++$index]->isGivenKind([T_STRING, T_NS_SEPARATOR, T_WHITESPACE])); + $className .= $tokens[$index]->getContent(); + } while ($tokens[++$index]->isGivenKind([T_STRING, T_NS_SEPARATOR])); return $className; } diff --git a/src/ClassNotation/OrderedOverridesFixer.php b/src/ClassNotation/OrderedOverridesFixer.php index 34fa3ea..927df09 100644 --- a/src/ClassNotation/OrderedOverridesFixer.php +++ b/src/ClassNotation/OrderedOverridesFixer.php @@ -4,21 +4,19 @@ namespace ErickSkrauch\PhpCsFixer\ClassNotation; use ErickSkrauch\PhpCsFixer\AbstractFixer; +use ErickSkrauch\PhpCsFixer\Analyzer\ClassElementsAnalyzer; use ErickSkrauch\PhpCsFixer\Analyzer\ClassNameAnalyzer; use PhpCsFixer\FixerDefinition\CodeSample; use PhpCsFixer\FixerDefinition\FixerDefinition; use PhpCsFixer\FixerDefinition\FixerDefinitionInterface; use PhpCsFixer\Tokenizer\Tokens; use ReflectionClass; +use ReflectionException; use SplFileInfo; use SplStack; /** - * @phpstan-type MethodData array{ - * name: string, - * start: int, - * end: int, - * } + * @phpstan-import-type AnalyzedClassElement from \ErickSkrauch\PhpCsFixer\Analyzer\ClassElementsAnalyzer */ final class OrderedOverridesFixer extends AbstractFixer { @@ -27,9 +25,15 @@ final class OrderedOverridesFixer extends AbstractFixer { */ private ClassNameAnalyzer $classNameAnalyzer; + /** + * @readonly + */ + private ClassElementsAnalyzer $classElementsAnalyzer; + public function __construct() { parent::__construct(); $this->classNameAnalyzer = new ClassNameAnalyzer(); + $this->classElementsAnalyzer = new ClassElementsAnalyzer(); } public function isCandidate(Tokens $tokens): bool { @@ -56,14 +60,12 @@ public function serialize() {} /** * Must run before OrderedClassElementsFixer * Must run after OrderedInterfacesFixer TODO: it's invariant right now: x < 0, but x > 65 + * see https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/issues/7760 */ public function getPriority(): int { return 75; } - /** - * @throws \ReflectionException - */ protected function applyFix(SplFileInfo $file, Tokens $tokens): void { for ($i = 1, $count = $tokens->count(); $i < $count; ++$i) { $classToken = $tokens[$i]; @@ -81,7 +83,12 @@ protected function applyFix(SplFileInfo $file, Tokens $tokens): void { } foreach ($extensions as $className) { - $classReflection = new ReflectionClass($className); + try { + $classReflection = new ReflectionClass($className); + } catch (ReflectionException $e) { // @phpstan-ignore catch.neverThrown + continue; + } + $parents = $this->getClassParents($classReflection, new SplStack()); foreach ($parents as $parent) { foreach ($parent->getMethods() as $method) { @@ -102,43 +109,16 @@ protected function applyFix(SplFileInfo $file, Tokens $tokens): void { $classBodyStart = $tokens->getNextTokenOfKind($i, ['{']); $classBodyEnd = $tokens->findBlockEnd(Tokens::BLOCK_TYPE_CURLY_BRACE, $classBodyStart); - /** @var list $unsortedMethods */ - $unsortedMethods = []; - // TODO: actually there still might be properties and traits in between methods declarations - for ($j = $classBodyStart; $j < $classBodyEnd; $j++) { - $functionToken = $tokens[$j]; - if (!$functionToken->isGivenKind(T_FUNCTION)) { - continue; - } - - $methodNameToken = $tokens[$tokens->getNextMeaningfulToken($j)]; - // Ensure it's not an anonymous function - if ($methodNameToken->equals('(')) { - continue; - } - - $methodName = $methodNameToken->getContent(); - - // Take the closest whitespace to the beginning of the method - $methodStart = $tokens->getPrevTokenOfKind($j, ['}', ';', '{']) + 1; - $methodEnd = $this->findElementEnd($tokens, $j); - - $unsortedMethods[] = [ - 'name' => $methodName, - 'start' => $methodStart, - 'end' => $methodEnd, - ]; - } - - $sortedMethods = $this->sortMethods($unsortedMethods, $methodsPriority); - if ($sortedMethods === $unsortedMethods) { + $unsortedElements = $this->classElementsAnalyzer->getClassElements($tokens, $classBodyStart); + $sortedElements = $this->sortElements($unsortedElements, $methodsPriority); + if ($sortedElements === $unsortedElements) { continue; } - $startIndex = $unsortedMethods[array_key_first($unsortedMethods)]['start']; - $endIndex = $unsortedMethods[array_key_last($unsortedMethods)]['end']; + $startIndex = $unsortedElements[array_key_first($unsortedElements)]['start']; + $endIndex = $unsortedElements[array_key_last($unsortedElements)]['end']; $replaceTokens = []; - foreach ($sortedMethods as $method) { + foreach ($sortedElements as $method) { for ($k = $method['start']; $k <= $method['end']; ++$k) { $replaceTokens[] = clone $tokens[$k]; } @@ -146,12 +126,12 @@ protected function applyFix(SplFileInfo $file, Tokens $tokens): void { $tokens->overrideRange($startIndex, $endIndex, $replaceTokens); - $i = $endIndex; + $i = $classBodyEnd + 1; } } /** - * @return array + * @return list */ private function getClassExtensions(Tokens $tokens, int $classTokenIndex, int $extensionType): array { $extensionTokenIndex = $tokens->getNextTokenOfKind($classTokenIndex, [[$extensionType], '{']); @@ -194,34 +174,20 @@ private function getClassParents(ReflectionClass $class, SplStack $stack): SplSt } /** - * Taken from the OrderedClassElementsFixer - */ - private function findElementEnd(Tokens $tokens, int $index): int { - $blockStart = $tokens->getNextTokenOfKind($index, ['{', ';']); - if ($tokens[$blockStart]->equals('{')) { - $blockEnd = $tokens->findBlockEnd(Tokens::BLOCK_TYPE_CURLY_BRACE, $blockStart); - } else { - $blockEnd = $blockStart; - } - - for (++$blockEnd; $tokens[$blockEnd]->isWhitespace(" \t") || $tokens[$blockEnd]->isComment(); ++$blockEnd); - - --$blockEnd; - - return $tokens[$blockEnd]->isWhitespace() ? $blockEnd - 1 : $blockEnd; - } - - /** - * @phpstan-param list $methods + * @phpstan-param list $elements * @phpstan-param array $methodsPriority * - * @phpstan-return list + * @phpstan-return list */ - private function sortMethods(array $methods, array $methodsPriority): array { - $count = count($methods); + private function sortElements(array $elements, array $methodsPriority): array { + $count = count($elements); $targetPriority = $methodsPriority[array_key_last($methodsPriority)]; for ($i = 0; $i < $count; $i++) { - $a = $methods[$i]; + $a = $elements[$i]; + if ($a['type'] !== 'method') { + continue; + } + if (!isset($methodsPriority[$a['name']])) { continue; } @@ -234,15 +200,19 @@ private function sortMethods(array $methods, array $methodsPriority): array { do { for ($j = $i + 1; $j < $count; $j++) { - $b = $methods[$j]; + $b = $elements[$j]; + if ($b['type'] !== 'method') { + continue; + } + if (!isset($methodsPriority[$b['name']])) { continue; } $priorityB = $methodsPriority[$b['name']]; if ($priorityB === $targetPriority) { - $methods[$i] = $b; - $methods[$j] = $a; + $elements[$i] = $b; + $elements[$j] = $a; $targetPriority--; continue 3; @@ -252,7 +222,7 @@ private function sortMethods(array $methods, array $methodsPriority): array { } // @phpstan-ignore return.type - return $methods; + return $elements; } } diff --git a/tests/Analyzer/ClassNameAnalyzerTest.php b/tests/Analyzer/ClassNameAnalyzerTest.php new file mode 100644 index 0000000..c656f2d --- /dev/null +++ b/tests/Analyzer/ClassNameAnalyzerTest.php @@ -0,0 +1,82 @@ +analyzer = new ClassNameAnalyzer(); + } + + /** + * @dataProvider provideValidCases + */ + public function testValid(string $code, int $index, string $expectedClassName): void { + $this->assertSame($expectedClassName, $this->analyzer->getFqn(Tokens::fromCode($code), $index)); + } + + /** + * @return iterable + */ + public function provideValidCases(): iterable { + yield 'of new simple' => [' [' [ + ' [ + ' [' [' [' [$multipleNamespacesCase, 13, '\A\DateTime']; + yield 'multiple namespaces B' => [$multipleNamespacesCase, 30, '\B\DateTime']; + yield 'multiple namespaces C' => [$multipleNamespacesCase, 49, '\C\DateTime']; + } + + public function testInvalid(): void { + $this->expectException(LogicException::class); + $this->expectExceptionMessage('No T_STRING or T_NS_SEPARATOR at given index 5, got "T_LNUMBER".'); + + $this->analyzer->getFqn(Tokens::fromCode(' [ + ' [ + ' [ + '