From f0265a012d9155c028c9a6aa765ac226409a6e94 Mon Sep 17 00:00:00 2001 From: Barney Laurance Date: Sat, 2 Dec 2023 11:50:50 +0000 Subject: [PATCH 1/2] 485: Include output from Psalm in report when mutant killed by static analysis --- .../Psalm/RunStaticAnalysisAgainstMutant.php | 27 ++++++++++++++----- .../RunStaticAnalysisAgainstEscapedMutant.php | 4 +-- .../RunStaticAnalysisAgainstMutantTest.php | 10 +++++++ ...StaticAnalysisAgainstEscapedMutantTest.php | 8 ++++-- 4 files changed, 39 insertions(+), 10 deletions(-) diff --git a/src/Roave/InfectionStaticAnalysis/Psalm/RunStaticAnalysisAgainstMutant.php b/src/Roave/InfectionStaticAnalysis/Psalm/RunStaticAnalysisAgainstMutant.php index 0a9c375..3db6384 100644 --- a/src/Roave/InfectionStaticAnalysis/Psalm/RunStaticAnalysisAgainstMutant.php +++ b/src/Roave/InfectionStaticAnalysis/Psalm/RunStaticAnalysisAgainstMutant.php @@ -5,13 +5,18 @@ namespace Roave\InfectionStaticAnalysis\Psalm; use Infection\Mutant\Mutant; +use Psalm\Internal\Analyzer\IssueData; use Psalm\Internal\Analyzer\ProjectAnalyzer; -use function array_key_exists; +use function array_map; use function count; +use function implode; /** * @internal + * @psalm-suppress InternalProperty - we use Psalm's internal IssueData class here. Afaik the only other way to + * display details of the issues would be to use one of the subclasses of \Psalm\Report, but I think none are exactly + * what we want. Probably we can accept the risk of Psalm's internals changing and breaking this. * * @final not explicitly final because we don't yet have a uniform API for this type of analysis */ @@ -19,12 +24,17 @@ class RunStaticAnalysisAgainstMutant { private bool $alreadyVisitedStubs = false; + /** @var IssueData[] */ + private array $psalmIssuesFromLastMutant = []; + public function __construct(private ProjectAnalyzer $projectAnalyzer) { } public function isMutantStillValidAccordingToStaticAnalysis(Mutant $mutant): bool { + $this->psalmIssuesFromLastMutant = []; + $path = $mutant->getFilePath(); $paths = [$mutant->getFilePath()]; $codebase = $this->projectAnalyzer->getCodebase(); @@ -48,13 +58,18 @@ public function isMutantStillValidAccordingToStaticAnalysis(Mutant $mutant): boo $codebase->reloadFiles($this->projectAnalyzer, $paths); $codebase->analyzer->analyzeFiles($this->projectAnalyzer, count($paths), false); - $mutationValid = ! array_key_exists( - $path, - $codebase->file_reference_provider->getExistingIssues(), - ); + $this->psalmIssuesFromLastMutant = $codebase->file_reference_provider->getExistingIssues()[$path] ?? []; $codebase->invalidateInformationForFile($path); - return $mutationValid; + return $this->psalmIssuesFromLastMutant === []; + } + + public function formatLastIssues(): string + { + return implode( + "\n\n", + array_map(static fn (IssueData $issueData) => ($issueData->type . ': ' . $issueData->message . "\n{$issueData->file_name}:{$issueData->line_from}"), $this->psalmIssuesFromLastMutant), + ); } } diff --git a/src/Roave/InfectionStaticAnalysis/RunStaticAnalysisAgainstEscapedMutant.php b/src/Roave/InfectionStaticAnalysis/RunStaticAnalysisAgainstEscapedMutant.php index 3d37dc9..be3068f 100644 --- a/src/Roave/InfectionStaticAnalysis/RunStaticAnalysisAgainstEscapedMutant.php +++ b/src/Roave/InfectionStaticAnalysis/RunStaticAnalysisAgainstEscapedMutant.php @@ -52,8 +52,8 @@ public function createFromProcess(MutantProcess $mutantProcess): MutantExecution assert(is_int($originalEndFilePosition)); return new MutantExecutionResult( - $result->getProcessCommandLine(), - $result->getProcessOutput(), + 'Static Analysis', + $this->runStaticAnalysis->formatLastIssues(), DetectionStatus::KILLED, // Mutant was squished by static analysis later(static fn () => yield $result->getMutantDiff()), $result->getMutantHash(), diff --git a/test/unit/Roave/InfectionStaticAnalysisTest/Psalm/RunStaticAnalysisAgainstMutantTest.php b/test/unit/Roave/InfectionStaticAnalysisTest/Psalm/RunStaticAnalysisAgainstMutantTest.php index 6dd6dc1..7290513 100644 --- a/test/unit/Roave/InfectionStaticAnalysisTest/Psalm/RunStaticAnalysisAgainstMutantTest.php +++ b/test/unit/Roave/InfectionStaticAnalysisTest/Psalm/RunStaticAnalysisAgainstMutantTest.php @@ -180,6 +180,16 @@ function add(int $a, int $b): int { } PHP, ))); + + self::assertStringContainsString( + "InvalidReturnStatement: The inferred type 'int' does not match the declared return type 'int<1, max>' for add", + $this->runStaticAnalysis->formatLastIssues(), + ); + + self::assertStringContainsString( + "InvalidReturnType: The declared return type 'int<1, max>' for add is incorrect, got 'int'", + $this->runStaticAnalysis->formatLastIssues(), + ); } public function testWillConsiderMutantReferencingProjectFilesAsValid(): void diff --git a/test/unit/Roave/InfectionStaticAnalysisTest/RunStaticAnalysisAgainstEscapedMutantTest.php b/test/unit/Roave/InfectionStaticAnalysisTest/RunStaticAnalysisAgainstEscapedMutantTest.php index 7183a33..71f3f9b 100644 --- a/test/unit/Roave/InfectionStaticAnalysisTest/RunStaticAnalysisAgainstEscapedMutantTest.php +++ b/test/unit/Roave/InfectionStaticAnalysisTest/RunStaticAnalysisAgainstEscapedMutantTest.php @@ -103,9 +103,13 @@ public function testWillKillMutantsThatEscapedAndFailedStaticAnalysis(): void ->method('isMutantStillValidAccordingToStaticAnalysis') ->willReturn(false); + $this->staticAnalysis->expects(self::any()) + ->method('formatLastIssues') + ->willReturn('formatted Psalm issues'); + $nextFactoryResult = new MutantExecutionResult( 'echo hi', - 'output', + 'formatted Psalm issues', DetectionStatus::ESCAPED, now('diff'), 'a-hash', @@ -136,7 +140,7 @@ public function testWillKillMutantsThatEscapedAndFailedStaticAnalysis(): void self::assertEquals($nextFactoryResult->getMutantDiff(), $result->getMutantDiff()); self::assertEquals($nextFactoryResult->getMutantHash(), $result->getMutantHash()); self::assertEquals($nextFactoryResult->getProcessOutput(), $result->getProcessOutput()); - self::assertEquals($nextFactoryResult->getProcessCommandLine(), $result->getProcessCommandLine()); + self::assertEquals('Static Analysis', $result->getProcessCommandLine()); self::assertSame(DetectionStatus::KILLED, $result->getDetectionStatus()); $reflectionOriginalStartFileLocation = new ReflectionProperty(MutantExecutionResult::class, 'originalStartFilePosition'); From 9d77deeb64f1f1b96c92b6cefcb9f359d091ba33 Mon Sep 17 00:00:00 2001 From: Barney Laurance Date: Sat, 2 Dec 2023 13:21:45 +0000 Subject: [PATCH 2/2] 485: Don't show file name or line number in report - not useful as file tested is temp file --- .../Psalm/RunStaticAnalysisAgainstMutant.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Roave/InfectionStaticAnalysis/Psalm/RunStaticAnalysisAgainstMutant.php b/src/Roave/InfectionStaticAnalysis/Psalm/RunStaticAnalysisAgainstMutant.php index 3db6384..4bade28 100644 --- a/src/Roave/InfectionStaticAnalysis/Psalm/RunStaticAnalysisAgainstMutant.php +++ b/src/Roave/InfectionStaticAnalysis/Psalm/RunStaticAnalysisAgainstMutant.php @@ -68,8 +68,8 @@ public function isMutantStillValidAccordingToStaticAnalysis(Mutant $mutant): boo public function formatLastIssues(): string { return implode( - "\n\n", - array_map(static fn (IssueData $issueData) => ($issueData->type . ': ' . $issueData->message . "\n{$issueData->file_name}:{$issueData->line_from}"), $this->psalmIssuesFromLastMutant), + ", ", + array_map(static fn (IssueData $issueData) => ($issueData->type . ': ' . $issueData->message), $this->psalmIssuesFromLastMutant), ); } }