From 7633ccf1b00b380a438362a53660e6c0e1b25e2c Mon Sep 17 00:00:00 2001 From: TomasVotruba Date: Sat, 10 Apr 2021 18:03:57 +0200 Subject: [PATCH] refactor AnnotationExtractor fomr FileDiff value object to standalone RectorsChangelogResolver service --- .../Annotation/AnnotationExtractorTest.php | 10 ++- .../RectorsChangelogResolverTest.php | 75 +++++++++++++++++++ .../Source/RectorWithChangelog.php | 4 +- .../Source/RectorWithOutChangelog.php | 4 +- .../ValueObject/FileDiffTest.php | 74 ------------------ .../RectorWithFileAndLineChangeTest.php | 43 ----------- .../Annotation/AnnotationExtractor.php | 9 +-- .../Annotation/RectorsChangelogResolver.php | 45 +++++++++++ .../Output/ConsoleOutputFormatter.php | 33 ++++++-- .../Output/JsonOutputFormatter.php | 14 +++- .../RectorWithFileAndLineChange.php | 23 +----- src/ValueObject/Reporting/FileDiff.php | 42 ++--------- 12 files changed, 180 insertions(+), 196 deletions(-) create mode 100644 packages-tests/ChangesReporting/Annotation/AppliedRectorsChangelogResolver/RectorsChangelogResolverTest.php rename packages-tests/ChangesReporting/{ValueObject => Annotation/AppliedRectorsChangelogResolver}/Source/RectorWithChangelog.php (82%) rename packages-tests/ChangesReporting/{ValueObject => Annotation/AppliedRectorsChangelogResolver}/Source/RectorWithOutChangelog.php (78%) delete mode 100644 packages-tests/ChangesReporting/ValueObject/FileDiffTest.php delete mode 100644 packages-tests/ChangesReporting/ValueObject/RectorWithFileAndLineChangeTest.php create mode 100644 packages/ChangesReporting/Annotation/RectorsChangelogResolver.php diff --git a/packages-tests/ChangesReporting/Annotation/AnnotationExtractorTest.php b/packages-tests/ChangesReporting/Annotation/AnnotationExtractorTest.php index 43c6dc8e7108..5eb306a9da67 100644 --- a/packages-tests/ChangesReporting/Annotation/AnnotationExtractorTest.php +++ b/packages-tests/ChangesReporting/Annotation/AnnotationExtractorTest.php @@ -6,8 +6,6 @@ use Iterator; use PHPUnit\Framework\TestCase; use Rector\ChangesReporting\Annotation\AnnotationExtractor; -use Rector\Tests\ChangesReporting\ValueObject\Source\RectorWithChangelog; -use Rector\Tests\ChangesReporting\ValueObject\Source\RectorWithOutChangelog; final class AnnotationExtractorTest extends TestCase { @@ -33,11 +31,15 @@ public function testExtractAnnotationFromClass(string $className, string $annota public function extractAnnotationProvider(): Iterator { yield 'Class with changelog annotation' => [ - RectorWithChangelog::class, + \Rector\Tests\ChangesReporting\Annotation\AppliedRectorsChangelogResolver\Source\RectorWithChangelog::class, '@changelog', 'https://github.com/rectorphp/rector/blob/master/docs/rector_rules_overview.md', ]; - yield 'Class without changelog annotation' => [RectorWithOutChangelog::class, '@changelog', null]; + yield 'Class without changelog annotation' => [ + \Rector\Tests\ChangesReporting\Annotation\AppliedRectorsChangelogResolver\Source\RectorWithOutChangelog::class, + '@changelog', + null, + ]; } } diff --git a/packages-tests/ChangesReporting/Annotation/AppliedRectorsChangelogResolver/RectorsChangelogResolverTest.php b/packages-tests/ChangesReporting/Annotation/AppliedRectorsChangelogResolver/RectorsChangelogResolverTest.php new file mode 100644 index 000000000000..3fbcbfe99190 --- /dev/null +++ b/packages-tests/ChangesReporting/Annotation/AppliedRectorsChangelogResolver/RectorsChangelogResolverTest.php @@ -0,0 +1,75 @@ +bootKernel(RectorKernel::class); + $this->rectorsChangelogResolver = $this->getService(RectorsChangelogResolver::class); + + $this->fileDiff = $this->createFileDiff(); + } + + public function test(): void + { + $rectorsChangelogs = $this->rectorsChangelogResolver->resolve($this->fileDiff->getRectorClasses()); + + $epectedRectorsChangelogs = [ + RectorWithChangelog::class => 'https://github.com/rectorphp/rector/blob/master/docs/rector_rules_overview.md', + ]; + $this->assertSame($epectedRectorsChangelogs, $rectorsChangelogs); + } + + private function createFileDiff(): FileDiff + { + // This is by intention to test the array_unique functionality + $rectorWithFileAndLineChange1 = new RectorWithFileAndLineChange( + new RectorWithChangelog(), + __DIR__ . '/Source/RectorWithChangelog.php', + 1 + ); + + $rectorWithFileAndLineChange2 = new RectorWithFileAndLineChange( + new RectorWithChangelog(), + __DIR__ . '/Source/RectorWithChangelog.php', + 1 + ); + + $rectorWithFileAndLineChange3 = new RectorWithFileAndLineChange( + new RectorWithOutChangelog(), + __DIR__ . '/Source/RectorWithOutChangelog.php', + 1 + ); + + $rectorWithFileAndLineChanges = [ + $rectorWithFileAndLineChange1, + $rectorWithFileAndLineChange2, + $rectorWithFileAndLineChange3, + ]; + + return new FileDiff(new SmartFileInfo(__FILE__), 'foo', 'foo', $rectorWithFileAndLineChanges); + } +} diff --git a/packages-tests/ChangesReporting/ValueObject/Source/RectorWithChangelog.php b/packages-tests/ChangesReporting/Annotation/AppliedRectorsChangelogResolver/Source/RectorWithChangelog.php similarity index 82% rename from packages-tests/ChangesReporting/ValueObject/Source/RectorWithChangelog.php rename to packages-tests/ChangesReporting/Annotation/AppliedRectorsChangelogResolver/Source/RectorWithChangelog.php index 653cf60f729e..4d5513218783 100644 --- a/packages-tests/ChangesReporting/ValueObject/Source/RectorWithChangelog.php +++ b/packages-tests/ChangesReporting/Annotation/AppliedRectorsChangelogResolver/Source/RectorWithChangelog.php @@ -1,8 +1,8 @@ createFileDiff(); - $this->assertSame([RectorWithChangelog::class, RectorWithOutChangelog::class], $fileDiff->getRectorClasses()); - } - - public function testGetRectorClassesWithChangelogUrlAndRectorClassAsKey(): void - { - $fileDiff = $this->createFileDiff(); - $this->assertSame( - [ - 'Rector\Tests\ChangesReporting\ValueObject\Source\RectorWithChangelog' => 'https://github.com/rectorphp/rector/blob/master/docs/rector_rules_overview.md', - ], - $fileDiff->getRectorClassesWithChangelogUrlAndRectorClassAsKey(new AnnotationExtractor()) - ); - } - - public function testGetRectorClassesWithChangelogUrl(): void - { - $fileDiff = $this->createFileDiff(); - $this->assertSame( - [ - 'Rector\Tests\ChangesReporting\ValueObject\Source\RectorWithChangelog (https://github.com/rectorphp/rector/blob/master/docs/rector_rules_overview.md)', - 'Rector\Tests\ChangesReporting\ValueObject\Source\RectorWithOutChangelog', - ], - $fileDiff->getRectorClassesWithChangelogUrl(new AnnotationExtractor()) - ); - } - - private function createFileDiff(): FileDiff - { - // This is by intention to test the array_unique functionality - $rectorWithFileAndLineChange1 = new RectorWithFileAndLineChange( - new RectorWithChangelog(), - __DIR__ . '/Source/RectorWithChangelog.php', - 1 - ); - - $rectorWithFileAndLineChange2 = new RectorWithFileAndLineChange( - new RectorWithChangelog(), - __DIR__ . '/Source/RectorWithChangelog.php', - 1 - ); - - $rectorWithFileAndLineChange3 = new RectorWithFileAndLineChange( - new RectorWithOutChangelog(), - __DIR__ . '/Source/RectorWithOutChangelog.php', - 1 - ); - - $rectorWithFileAndLineChanges = [ - $rectorWithFileAndLineChange1, - $rectorWithFileAndLineChange2, - $rectorWithFileAndLineChange3, - ]; - - return new FileDiff(new SmartFileInfo(__FILE__), 'foo', 'foo', $rectorWithFileAndLineChanges); - } -} diff --git a/packages-tests/ChangesReporting/ValueObject/RectorWithFileAndLineChangeTest.php b/packages-tests/ChangesReporting/ValueObject/RectorWithFileAndLineChangeTest.php deleted file mode 100644 index efdda477331e..000000000000 --- a/packages-tests/ChangesReporting/ValueObject/RectorWithFileAndLineChangeTest.php +++ /dev/null @@ -1,43 +0,0 @@ -assertSame( - $expected, - $rectorWithFileAndLineChange->getRectorClassWithChangelogUrl(new AnnotationExtractor()) - ); - } - - /** - * @return Iterator>> - */ - public function rectorsWithFileAndLineChange(): Iterator - { - yield 'Rector with link' => [ - 'Rector\Tests\ChangesReporting\ValueObject\Source\RectorWithChangelog (https://github.com/rectorphp/rector/blob/master/docs/rector_rules_overview.md)', - new RectorWithFileAndLineChange(new RectorWithChangelog(), __DIR__ . '/Source/RectorWithLink.php', 1), - ]; - - yield 'Rector without link' => [ - 'Rector\Tests\ChangesReporting\ValueObject\Source\RectorWithOutChangelog', - new RectorWithFileAndLineChange(new RectorWithOutChangelog(), __DIR__ . '/Source/RectorWithLink.php', 1), - ]; - } -} diff --git a/packages/ChangesReporting/Annotation/AnnotationExtractor.php b/packages/ChangesReporting/Annotation/AnnotationExtractor.php index 686826669b4a..c7eb67b18d3d 100644 --- a/packages/ChangesReporting/Annotation/AnnotationExtractor.php +++ b/packages/ChangesReporting/Annotation/AnnotationExtractor.php @@ -1,9 +1,11 @@ $className + * @param class-string $className */ public function extractAnnotationFromClass(string $className, string $annotation): ?string { @@ -25,11 +27,8 @@ public function extractAnnotationFromClass(string $className, string $annotation // @see https://regex101.com/r/oYGaWU/1 $pattern = '#' . preg_quote($annotation, '#') . '\s+(?.*?)$#m'; - $matches = Strings::match($docComment, $pattern); - if ($matches === false) { - return null; - } + $matches = Strings::match($docComment, $pattern); return $matches['content'] ?? null; } } diff --git a/packages/ChangesReporting/Annotation/RectorsChangelogResolver.php b/packages/ChangesReporting/Annotation/RectorsChangelogResolver.php new file mode 100644 index 000000000000..58924de97a42 --- /dev/null +++ b/packages/ChangesReporting/Annotation/RectorsChangelogResolver.php @@ -0,0 +1,45 @@ +annotationExtractor = $annotationExtractor; + } + + /** + * @param array> $rectorClasses + * @return array + */ + public function resolve(array $rectorClasses): array + { + $rectorClassesToChangelogUrls = $this->resolveIncludingMissing($rectorClasses); + return array_filter($rectorClassesToChangelogUrls); + } + + /** + * @param array> $rectorClasses + * @return array + */ + public function resolveIncludingMissing(array $rectorClasses): array + { + $rectorClassesToChangelogUrls = []; + foreach ($rectorClasses as $rectorClass) { + $changelogUrl = $this->annotationExtractor->extractAnnotationFromClass($rectorClass, '@changelog'); + $rectorClassesToChangelogUrls[$rectorClass] = $changelogUrl; + } + + return $rectorClassesToChangelogUrls; + } +} diff --git a/packages/ChangesReporting/Output/ConsoleOutputFormatter.php b/packages/ChangesReporting/Output/ConsoleOutputFormatter.php index 8fdbb236baf1..b749f81a7651 100644 --- a/packages/ChangesReporting/Output/ConsoleOutputFormatter.php +++ b/packages/ChangesReporting/Output/ConsoleOutputFormatter.php @@ -5,7 +5,7 @@ namespace Rector\ChangesReporting\Output; use Nette\Utils\Strings; -use Rector\ChangesReporting\Annotation\AnnotationExtractor; +use Rector\ChangesReporting\Annotation\RectorsChangelogResolver; use Rector\ChangesReporting\Application\ErrorAndDiffCollector; use Rector\ChangesReporting\Contract\Output\OutputFormatterInterface; use Rector\Core\Configuration\Configuration; @@ -46,20 +46,20 @@ final class ConsoleOutputFormatter implements OutputFormatterInterface private $betterStandardPrinter; /** - * @var AnnotationExtractor + * @var RectorsChangelogResolver */ - private $annotationExtractor; + private $rectorsChangelogResolver; public function __construct( BetterStandardPrinter $betterStandardPrinter, Configuration $configuration, SymfonyStyle $symfonyStyle, - AnnotationExtractor $annotationExtractor + RectorsChangelogResolver $rectorsChangelogResolver ) { $this->symfonyStyle = $symfonyStyle; $this->betterStandardPrinter = $betterStandardPrinter; $this->configuration = $configuration; - $this->annotationExtractor = $annotationExtractor; + $this->rectorsChangelogResolver = $rectorsChangelogResolver; } public function report(ErrorAndDiffCollector $errorAndDiffCollector): void @@ -119,10 +119,12 @@ private function reportFileDiffs(array $fileDiffs): void $this->symfonyStyle->writeln($fileDiff->getDiffConsoleFormatted()); $this->symfonyStyle->newLine(); + $rectorsChangelogsLines = $this->createRectorChangelogLines($fileDiff); + if ($fileDiff->getRectorChanges() !== []) { $this->symfonyStyle->writeln('Applied rules:'); $this->symfonyStyle->newLine(); - $this->symfonyStyle->listing($fileDiff->getRectorClassesWithChangelogUrl($this->annotationExtractor)); + $this->symfonyStyle->listing($rectorsChangelogsLines); $this->symfonyStyle->newLine(); } } @@ -232,4 +234,23 @@ private function createSuccessMessage(ErrorAndDiffCollector $errorAndDiffCollect $this->configuration->isDryRun() ? 'would have changed (dry-run)' : ($changeCount === 1 ? 'has' : 'have') . ' been changed' ); } + + /** + * @return string[] + */ + private function createRectorChangelogLines(FileDiff $fileDiff): array + { + $rectorsChangelogs = $this->rectorsChangelogResolver->resolveIncludingMissing($fileDiff->getRectorClasses()); + + $rectorsChangelogsLines = []; + foreach ($rectorsChangelogs as $rectorClass => $changelog) { + if ($changelog === null) { + $rectorsChangelogsLines[] = $rectorClass; + } else { + $rectorsChangelogsLines[] = $rectorClass . ' ' . $changelog; + } + } + + return $rectorsChangelogsLines; + } } diff --git a/packages/ChangesReporting/Output/JsonOutputFormatter.php b/packages/ChangesReporting/Output/JsonOutputFormatter.php index a5a44d431905..f6e062333156 100644 --- a/packages/ChangesReporting/Output/JsonOutputFormatter.php +++ b/packages/ChangesReporting/Output/JsonOutputFormatter.php @@ -6,6 +6,7 @@ use Nette\Utils\Json; use Rector\ChangesReporting\Annotation\AnnotationExtractor; +use Rector\ChangesReporting\Annotation\RectorsChangelogResolver; use Rector\ChangesReporting\Application\ErrorAndDiffCollector; use Rector\ChangesReporting\Contract\Output\OutputFormatterInterface; use Rector\Core\Configuration\Configuration; @@ -33,14 +34,21 @@ final class JsonOutputFormatter implements OutputFormatterInterface */ private $annotationExtractor; + /** + * @var RectorsChangelogResolver + */ + private $appliedRectorsWithChangelogResolver; + public function __construct( Configuration $configuration, SmartFileSystem $smartFileSystem, - AnnotationExtractor $annotationExtractor + AnnotationExtractor $annotationExtractor, + RectorsChangelogResolver $appliedRectorsWithChangelogResolver ) { $this->configuration = $configuration; $this->smartFileSystem = $smartFileSystem; $this->annotationExtractor = $annotationExtractor; + $this->appliedRectorsWithChangelogResolver = $appliedRectorsWithChangelogResolver; } public function getName(): string @@ -67,8 +75,8 @@ public function report(ErrorAndDiffCollector $errorAndDiffCollector): void foreach ($fileDiffs as $fileDiff) { $relativeFilePath = $fileDiff->getRelativeFilePath(); - $appliedRectorsWithChangelog = $fileDiff->getRectorClassesWithChangelogUrlAndRectorClassAsKey( - $this->annotationExtractor + $appliedRectorsWithChangelog = $this->appliedRectorsWithChangelogResolver->resolve( + $fileDiff->getRectorClasses() ); $errorsArray['file_diffs'][] = [ diff --git a/packages/ChangesReporting/ValueObject/RectorWithFileAndLineChange.php b/packages/ChangesReporting/ValueObject/RectorWithFileAndLineChange.php index 21eb898fc39b..9aceabb04fcd 100644 --- a/packages/ChangesReporting/ValueObject/RectorWithFileAndLineChange.php +++ b/packages/ChangesReporting/ValueObject/RectorWithFileAndLineChange.php @@ -4,7 +4,6 @@ namespace Rector\ChangesReporting\ValueObject; -use Rector\ChangesReporting\Annotation\AnnotationExtractor; use Rector\Core\Contract\Rector\RectorInterface; final class RectorWithFileAndLineChange @@ -37,30 +36,14 @@ public function getRectorDefinitionsDescription(): string return $ruleDefinition->getDescription(); } + /** + * @return class-string + */ public function getRectorClass(): string { return get_class($this->rector); } - public function getRectorClassWithChangelogUrl(AnnotationExtractor $annotationExtractor): string - { - $rectorClass = get_class($this->rector); - $changeLogUrl = $this->getChangelogUrl($annotationExtractor); - - if ($changeLogUrl === null) { - return $rectorClass; - } - - return sprintf('%s (%s)', $rectorClass, $changeLogUrl); - } - - public function getChangelogUrl(AnnotationExtractor $annotationExtractor): ?string - { - $rectorClass = get_class($this->rector); - - return $annotationExtractor->extractAnnotationFromClass($rectorClass, '@changelog'); - } - public function getLine(): int { return $this->line; diff --git a/src/ValueObject/Reporting/FileDiff.php b/src/ValueObject/Reporting/FileDiff.php index aad994684258..aa9180fd9850 100644 --- a/src/ValueObject/Reporting/FileDiff.php +++ b/src/ValueObject/Reporting/FileDiff.php @@ -4,8 +4,8 @@ namespace Rector\Core\ValueObject\Reporting; -use Rector\ChangesReporting\Annotation\AnnotationExtractor; use Rector\ChangesReporting\ValueObject\RectorWithFileAndLineChange; +use Rector\Core\Contract\Rector\RectorInterface; use Symplify\SmartFileSystem\SmartFileInfo; final class FileDiff @@ -74,7 +74,7 @@ public function getRectorChanges(): array } /** - * @return string[] + * @return array> */ public function getRectorClasses(): array { @@ -87,45 +87,13 @@ public function getRectorClasses(): array } /** - * @return string[] - */ - public function getRectorClassesWithChangelogUrl(AnnotationExtractor $annotationExtractor): array - { - $rectorClasses = []; - foreach ($this->rectorWithFileAndLineChanges as $rectorWithFileAndLineChange) { - $rectorClasses[] = $rectorWithFileAndLineChange->getRectorClassWithChangelogUrl($annotationExtractor); - } - - return $this->sortClasses($rectorClasses); - } - - /** - * @return array - */ - public function getRectorClassesWithChangelogUrlAndRectorClassAsKey(AnnotationExtractor $annotationExtractor): array - { - $rectorClasses = []; - foreach ($this->rectorWithFileAndLineChanges as $rectorWithFileAndLineChange) { - $changelogUrl = $rectorWithFileAndLineChange->getChangelogUrl($annotationExtractor); - if ($changelogUrl !== null) { - $rectorClasses[$rectorWithFileAndLineChange->getRectorClass()] = $changelogUrl; - } - } - $rectorClasses = array_unique($rectorClasses); - - ksort($rectorClasses); - - return $rectorClasses; - } - - /** - * @param string[] $rectorClasses - * @return string[] + * @template TType as object + * @param array> $rectorClasses + * @return array> */ private function sortClasses(array $rectorClasses): array { $rectorClasses = array_unique($rectorClasses); - sort($rectorClasses); return $rectorClasses;