diff options
author | jenkins-bot <jenkins-bot@gerrit.wikimedia.org> | 2024-11-07 16:53:51 +0000 |
---|---|---|
committer | Gerrit Code Review <gerrit@wikimedia.org> | 2024-11-07 16:53:51 +0000 |
commit | 9e07f6ccbe9d1a7ef63efa705d57beb04c4c81a4 (patch) | |
tree | d13cc6b5b483261614dff23bc79afc32d512ef94 /includes/composer/PhpUnitSplitter | |
parent | fc98265c6a964d149d6cecb3e17edc4e97fccf56 (diff) | |
parent | 548354555b7842ec9b0173bd42a6d4a72c65c8a1 (diff) | |
download | mediawikicore-9e07f6ccbe9d1a7ef63efa705d57beb04c4c81a4.tar.gz mediawikicore-9e07f6ccbe9d1a7ef63efa705d57beb04c4c81a4.zip |
Merge "Collect test failure logs and print them at the end of parallel runs"
Diffstat (limited to 'includes/composer/PhpUnitSplitter')
4 files changed, 479 insertions, 0 deletions
diff --git a/includes/composer/PhpUnitSplitter/PhpUnitConsoleOutputProcessingException.php b/includes/composer/PhpUnitSplitter/PhpUnitConsoleOutputProcessingException.php new file mode 100644 index 000000000000..d83dbb866984 --- /dev/null +++ b/includes/composer/PhpUnitSplitter/PhpUnitConsoleOutputProcessingException.php @@ -0,0 +1,9 @@ +<?php +declare( strict_types = 1 ); + +namespace MediaWiki\Composer\PhpUnitSplitter; + +use Exception; + +class PhpUnitConsoleOutputProcessingException extends Exception { +} diff --git a/includes/composer/PhpUnitSplitter/PhpUnitConsoleOutputProcessor.php b/includes/composer/PhpUnitSplitter/PhpUnitConsoleOutputProcessor.php new file mode 100644 index 000000000000..49d50a79ee9b --- /dev/null +++ b/includes/composer/PhpUnitSplitter/PhpUnitConsoleOutputProcessor.php @@ -0,0 +1,361 @@ +<?php + +declare( strict_types = 1 ); + +namespace MediaWiki\Composer\PhpUnitSplitter; + +use Composer\IO\IOInterface; + +/** + * @license GPL-2.0-or-later + */ +class PhpUnitConsoleOutputProcessor { + + private const STATE_EXPECT_PHP_VERSION = 0; + private const STATE_EXPECT_PHPUNIT_VERSION = 1; + private const STATE_EXPECT_DOT_CHART = 2; + private const STATE_EXPECT_TEST_SUMMARY = 3; + private const STATE_EXPECT_ERROR_SUMMARY = 4; + private const STATE_EXPECT_FAILURE_SUMMARY = 5; + private const STATE_EXPECT_ERROR_TOTALS = 6; + private const STATE_EXPECT_SLOW_TESTS = 7; + private const STATE_EOF = 8; + private const STATE_CLOSED = 9; + + private int $state = self::STATE_EXPECT_PHP_VERSION; + private array $failures = []; + private array $errors = []; + private array $slowTests = []; + private ?string $phpVersion = null; + private ?string $phpUnitVersion = null; + private ?string $dotChart = null; + private bool $noTestsExecuted = false; + private ?PhpUnitFailure $currentFailure = null; + private int $testCount = 0; + private int $assertionCount = 0; + private int $errorCount = 0; + private int $failureCount = 0; + private int $skippedCount = 0; + + public static function writeOutputToLogFile( + string $logFilename, ?string $consoleOutput + ) { + if ( !$consoleOutput ) { + return; + } + file_put_contents( $logFilename, $consoleOutput ); + } + + /** + * @throws PhpUnitConsoleOutputProcessingException + */ + public function processInput( string $data ): void { + $array = preg_split( "/\r\n|\n|\r/", $data ); + foreach ( $array as $inputLine ) { + $matches = []; + switch ( $this->state ) { + case self::STATE_EXPECT_PHP_VERSION: + if ( preg_match( "/^Using PHP (.*)$/", $inputLine, $matches ) ) { + $this->phpVersion = $matches[1]; + $this->state = self::STATE_EXPECT_PHPUNIT_VERSION; + } + break; + + case self::STATE_EXPECT_PHPUNIT_VERSION: + if ( preg_match( "/^PHPUnit (.*) by .*$/", $inputLine, $matches ) ) { + $this->phpUnitVersion = $matches[1]; + $this->state = self::STATE_EXPECT_DOT_CHART; + } + break; + + case self::STATE_EXPECT_DOT_CHART: + $this->handleDotChartLine( $inputLine ); + break; + + case self::STATE_EXPECT_TEST_SUMMARY: + $this->handleTestSummary( $inputLine ); + break; + + case self::STATE_EXPECT_ERROR_SUMMARY: + $this->handleSummaryTotals( $inputLine, false ); + break; + + case self::STATE_EXPECT_FAILURE_SUMMARY: + $this->handleSummaryTotals( $inputLine, true ); + break; + + case self::STATE_EXPECT_ERROR_TOTALS: + $this->processPossibleErrorTotalsLine( $inputLine ); + break; + + case self::STATE_EXPECT_SLOW_TESTS: + if ( preg_match( "/^ \d+\. (\d+)ms to run (.*)$/", $inputLine, $matches ) ) { + $this->slowTests[] = new PhpUnitSlowTest( intval( $matches[1] ), $matches[2] ); + } + break; + + case self::STATE_EOF: + if ( $inputLine ) { + throw new PhpUnitConsoleOutputProcessingException( + "Unexpected input in `EOF` state: '" . $inputLine . "'" + ); + } + break; + + default: + throw new PhpUnitConsoleOutputProcessingException( + "Unexpected processing state " . $this->state + ); + } + } + if ( $this->currentFailure && !$this->currentFailure->empty() ) { + $this->failures[] = $this->currentFailure; + } + } + + /** + * @throws PhpUnitConsoleOutputProcessingException + */ + private function handleSummaryTotals( string $inputLine, bool $errorSectionComplete ) { + if ( preg_match( "/^.*ERRORS!.*$/", $inputLine ) || + preg_match( "/^.*FAILURES!.*$/", $inputLine ) ) { + $this->state = self::STATE_EXPECT_ERROR_TOTALS; + return; + } + if ( !$errorSectionComplete && ( + preg_match( "/^There were (\d+) failures:$/", $inputLine, $matches ) || + $inputLine === "There was 1 failure:" ) + ) { + if ( $this->currentFailure && !$this->currentFailure->empty() ) { + $this->errors[] = $this->currentFailure; + } + $this->state = self::STATE_EXPECT_FAILURE_SUMMARY; + $this->currentFailure = new PhpUnitFailure(); + return; + } + if ( $this->processPossibleErrorTotalsLine( $inputLine ) ) { + return; + } + if ( !$this->currentFailure->processLine( $inputLine ) ) { + if ( !$errorSectionComplete ) { + $this->errors[] = $this->currentFailure; + } else { + $this->failures[] = $this->currentFailure; + } + $this->currentFailure = new PhpUnitFailure(); + $this->currentFailure->processLine( $inputLine ); + } + } + + private function handleDotChartLine( string $inputLine ) { + if ( $this->dotChart === null ) { + $this->dotChart = ""; + } + $this->dotChart .= $inputLine . PHP_EOL; + if ( preg_match( "/^.* \(100%\)$/", $inputLine ) ) { + $this->state = self::STATE_EXPECT_TEST_SUMMARY; + return; + } + if ( preg_match( "/^.*No tests executed!.*$/", $inputLine ) ) { + $this->noTestsExecuted = true; + $this->state = self::STATE_EOF; + } + } + + private function handleTestSummarySection( string $inputLine, string $keyword, int $nextState ): bool { + if ( preg_match( "/^There were (\d+) " . $keyword . "s:$/", $inputLine ) || + $inputLine === "There was 1 " . $keyword . ":" ) { + $this->state = $nextState; + $this->currentFailure = new PhpUnitFailure(); + return true; + } + return false; + } + + private function handleTestSummary( string $inputLine ) { + if ( $this->handleTestSummarySection( + $inputLine, + "error", + self::STATE_EXPECT_ERROR_SUMMARY + ) ) { + return; + } + if ( $this->handleTestSummarySection( + $inputLine, + "failure", + self::STATE_EXPECT_FAILURE_SUMMARY + ) ) { + return; + } + $this->processPossibleErrorTotalsLine( $inputLine ); + } + + private function processPossibleErrorTotalsLine( string $inputLine ): bool { + $matches = []; + if ( preg_match( "/^.*Tests: (\d+).*$/", $inputLine, $matches ) ) { + $this->testCount = intval( $matches[1] ); + if ( preg_match( "/^.*Assertions: (\d+).*$/", $inputLine, $matches ) ) { + $this->assertionCount = intval( $matches[1] ); + } + if ( preg_match( "/^.*Failures: (\d+).*$/", $inputLine, $matches ) ) { + $this->failureCount = intval( $matches[1] ); + } + if ( preg_match( "/^.*Errors: (\d+).*$/", $inputLine, $matches ) ) { + $this->errorCount = intval( $matches[1] ); + } + if ( preg_match( "/^.*Skipped: (\d+).*$/", $inputLine, $matches ) ) { + $this->skippedCount = intval( $matches[1] ); + } + $this->state = self::STATE_EXPECT_SLOW_TESTS; + return true; + } + if ( preg_match( "/^.*OK \((\d+) tests?, (\d+) assertions?\).*$/", $inputLine, $matches ) ) { + $this->testCount = intval( $matches[1] ); + $this->assertionCount = intval( $matches[2] ); + $this->state = self::STATE_EXPECT_SLOW_TESTS; + } + return false; + } + + /** + * @throws PhpUnitConsoleOutputProcessingException + */ + public static function collectAndDumpFailureSummary( string $filePattern, int $groupCount, IOInterface $io ): bool { + $failuresFound = false; + $slowTests = []; + for ( $i = 0; $i < $groupCount; $i++ ) { + $filename = sprintf( $filePattern, $i ); + if ( file_exists( $filename ) ) { + $summary = new PhpUnitConsoleOutputProcessor(); + $summary->processInput( file_get_contents( $filename ) ); + $summary->close(); + $slowTests = array_values( array_merge( $slowTests, $summary->getSlowTests() ) ); + $failureDetails = $summary->getFailureDetails(); + if ( $failureDetails ) { + $io->write( "Report from `split_group" . $i . "`:" . PHP_EOL ); + $io->write( $failureDetails ); + $failuresFound = true; + } + } + } + if ( count( $slowTests ) > 0 ) { + $io->write( PHP_EOL . "You should really speed up these slow tests (>100ms)..." . PHP_EOL ); + usort( $slowTests, static fn ( $t1, $t2 ) => $t2->getDuration() - $t1->getDuration() ); + for ( $i = 0; $i < min( 10, count( $slowTests ) ); $i++ ) { + $test = $slowTests[$i]; + $io->write( " " . ( $i + 1 ) . ". " . $test->getDuration() . "ms to run " . $test->getTest() ); + } + } + return $failuresFound; + } + + /** + * @throws PhpUnitConsoleOutputProcessingException + */ + public function getPhpVersion(): string { + if ( $this->state !== self::STATE_CLOSED ) { + throw new PhpUnitConsoleOutputProcessingException( "Still processing. Call `close()` first" ); + } + if ( !$this->phpVersion ) { + throw new PhpUnitConsoleOutputProcessingException( "No php version string detceted" ); + } + return $this->phpVersion; + } + + /** + * @throws PhpUnitConsoleOutputProcessingException + */ + public function getPhpUnitVersion(): string { + if ( $this->state !== self::STATE_CLOSED ) { + throw new PhpUnitConsoleOutputProcessingException( "Still processing. Call `close()` first" ); + } + if ( !$this->phpUnitVersion ) { + throw new PhpUnitConsoleOutputProcessingException( "No phpunit version string detceted" ); + } + return $this->phpUnitVersion; + } + + /** + * @throws PhpUnitConsoleOutputProcessingException + */ + public function wereTestsExecuted(): bool { + if ( $this->state !== self::STATE_CLOSED ) { + throw new PhpUnitConsoleOutputProcessingException( "Still processing. Call `close()` first" ); + } + return !$this->noTestsExecuted; + } + + public function close(): void { + $this->state = self::STATE_CLOSED; + } + + /** + * @throws PhpUnitConsoleOutputProcessingException + */ + public function hasFailures(): bool { + if ( $this->state !== self::STATE_CLOSED ) { + throw new PhpUnitConsoleOutputProcessingException( "Still processing. Call `close()` first" ); + } + return count( $this->failures ) + count( $this->errors ) > 0; + } + + public function getFailureDetails(): string { + $errorDetails = $this->prettyPrintErrors(); + $failureDetails = $this->prettyPrintFailures(); + $joiner = ""; + if ( $errorDetails && $failureDetails ) { + $joiner = PHP_EOL . "--" . PHP_EOL . PHP_EOL; + } + return $errorDetails . $joiner . $failureDetails; + } + + public function getSlowTests(): array { + return $this->slowTests; + } + + private function prettyPrintErrors(): string { + $errorCount = count( $this->errors ); + if ( $errorCount === 0 ) { + return ""; + } + $result = "There " . ( $errorCount > 1 ? "were " : "was " ) . $errorCount + . " error" . ( $errorCount > 1 ? "s" : "" ) . ":" . PHP_EOL . PHP_EOL; + return $result . implode( + PHP_EOL, + array_map( static fn ( $err ) => $err->getFailureDetails(), array_merge( $this->errors ) ) + ); + } + + private function prettyPrintFailures(): string { + $failureCount = count( $this->failures ); + if ( $failureCount === 0 ) { + return ""; + } + $result = "There " . ( $failureCount > 1 ? "were " : "was " ) . $failureCount + . " failure" . ( $failureCount > 1 ? "s" : "" ) . ":" . PHP_EOL . PHP_EOL; + return $result . implode( + PHP_EOL, + array_map( static fn ( $err ) => $err->getFailureDetails(), array_merge( $this->failures ) ) + ); + } + + public function getAssertionCount(): int { + return $this->assertionCount; + } + + public function getErrorCount(): int { + return $this->errorCount; + } + + public function getFailureCount(): int { + return $this->failureCount; + } + + public function getTestCount(): int { + return $this->testCount; + } + + public function getSkippedCount(): int { + return $this->skippedCount; + } +} diff --git a/includes/composer/PhpUnitSplitter/PhpUnitFailure.php b/includes/composer/PhpUnitSplitter/PhpUnitFailure.php new file mode 100644 index 000000000000..3b398b04d2c7 --- /dev/null +++ b/includes/composer/PhpUnitSplitter/PhpUnitFailure.php @@ -0,0 +1,85 @@ +<?php +declare( strict_types = 1 ); + +namespace MediaWiki\Composer\PhpUnitSplitter; + +class PhpUnitFailure { + + private const STATE_EXPECT_TEST_ID_AND_NAME = 0; + private const STATE_EXPECT_FAILURE_DETAIL = 1; + private const STATE_EXPECT_LOGS = 2; + + private int $failureNumber = 0; + private string $testCase; + private ?string $dataSet = null; + private ?string $failureDetail; + private ?string $logs; + private int $state = self::STATE_EXPECT_TEST_ID_AND_NAME; + + public function getFailureDetails(): ?string { + $result = $this->failureNumber . ") " . $this->testCase; + if ( $this->dataSet ) { + $result .= " with data set " . $this->dataSet; + } + $result .= PHP_EOL; + $result .= $this->failureDetail; + return $result; + } + + public function empty(): bool { + return $this->failureNumber === 0; + } + + /** + * @throws PhpUnitConsoleOutputProcessingException + */ + public function processLine( string $line ): bool { + $matches = []; + switch ( $this->state ) { + case self::STATE_EXPECT_TEST_ID_AND_NAME: + if ( preg_match( + "/^(\d+)\) (.*::[^\b]+)( with data set (.*))?$/", + $line, + $matches, + PREG_UNMATCHED_AS_NULL + ) ) { + $this->failureNumber = intval( $matches[1] ); + $this->testCase = $matches[2]; + if ( $matches[3] !== null ) { + $this->dataSet = $matches[4]; + } + $this->state = self::STATE_EXPECT_FAILURE_DETAIL; + $this->failureDetail = ""; + } + break; + + case self::STATE_EXPECT_FAILURE_DETAIL: + if ( $line === "=== Logs generated by test case" ) { + $this->state = self::STATE_EXPECT_LOGS; + $this->logs = ""; + break; + } + if ( preg_match( "/^(\d+)\) (.*::[^\b]+)/", $line ) ) { + // Start of next error case + return false; + } + $this->failureDetail .= $line . PHP_EOL; + break; + + case self::STATE_EXPECT_LOGS: + if ( preg_match( "/^(\d+)\) (.*::[^\b]+)/", $line ) || + $line === "===" ) { + // Start of next error case + return false; + } + $this->logs .= $line . PHP_EOL; + break; + + default: + throw new PhpUnitConsoleOutputProcessingException( + "Unexpected processing state " . $this->state + ); + } + return true; + } +} diff --git a/includes/composer/PhpUnitSplitter/PhpUnitSlowTest.php b/includes/composer/PhpUnitSplitter/PhpUnitSlowTest.php new file mode 100644 index 000000000000..a0be76f2e917 --- /dev/null +++ b/includes/composer/PhpUnitSplitter/PhpUnitSlowTest.php @@ -0,0 +1,24 @@ +<?php +declare( strict_types = 1 ); + +namespace MediaWiki\Composer\PhpUnitSplitter; + +class PhpUnitSlowTest { + + private int $duration; + private string $test; + + public function __construct( int $duration, string $test ) { + $this->duration = $duration; + $this->test = $test; + } + + public function getDuration(): int { + return $this->duration; + } + + public function getTest(): string { + return $this->test; + } + +} |