mirror of
https://github.com/pestphp/pest.git
synced 2026-03-07 00:07:22 +01:00
337 lines
9.5 KiB
PHP
337 lines
9.5 KiB
PHP
<?php
|
||
|
||
declare(strict_types=1);
|
||
|
||
namespace Pest\Logging;
|
||
|
||
use function getmypid;
|
||
use Pest\Concerns\Testable;
|
||
use function Pest\version;
|
||
use PHPUnit\Framework\AssertionFailedError;
|
||
use PHPUnit\Framework\Test;
|
||
use PHPUnit\Framework\TestCase;
|
||
use PHPUnit\Framework\TestResult;
|
||
use PHPUnit\Framework\TestSuite;
|
||
use PHPUnit\Framework\Warning;
|
||
use PHPUnit\Runner\PhptTestCase;
|
||
use PHPUnit\TextUI\DefaultResultPrinter;
|
||
use function round;
|
||
use function str_replace;
|
||
use function strlen;
|
||
use Throwable;
|
||
|
||
final class TeamCity extends DefaultResultPrinter
|
||
{
|
||
private const PROTOCOL = 'pest_qn://';
|
||
private const NAME = 'name';
|
||
private const LOCATION_HINT = 'locationHint';
|
||
private const DURATION = 'duration';
|
||
private const TEST_SUITE_STARTED = 'testSuiteStarted';
|
||
private const TEST_SUITE_FINISHED = 'testSuiteFinished';
|
||
|
||
/** @var int */
|
||
private $flowId;
|
||
|
||
/** @var bool */
|
||
private $isSummaryTestCountPrinted = false;
|
||
|
||
/** @var \PHPUnit\Util\Log\TeamCity */
|
||
private $phpunitTeamCity;
|
||
|
||
/**
|
||
* @param resource|string|null $out
|
||
*/
|
||
public function __construct($out, bool $verbose, string $colors)
|
||
{
|
||
parent::__construct($out, $verbose, $colors, false, 80, false);
|
||
$this->phpunitTeamCity = new \PHPUnit\Util\Log\TeamCity(
|
||
$out,
|
||
$verbose,
|
||
$colors,
|
||
false,
|
||
80,
|
||
false
|
||
);
|
||
|
||
$this->logo();
|
||
}
|
||
|
||
private function logo(): void
|
||
{
|
||
$this->writeNewLine();
|
||
$this->write('Pest '.version());
|
||
$this->writeNewLine();
|
||
}
|
||
|
||
public function printResult(TestResult $result): void
|
||
{
|
||
$this->printFooter($result);
|
||
}
|
||
|
||
protected function printFooter(TestResult $result): void
|
||
{
|
||
$this->writeProgress('Tests: ');
|
||
|
||
$results = [
|
||
'failed' => ['count' => $result->errorCount() + $result->failureCount(), 'color' => 'fg-red'],
|
||
'skipped' => ['count' => $result->skippedCount(), 'color' => 'fg-cyan'],
|
||
'warned' => ['count' => $result->warningCount(), 'color' => 'fg-cyan'],
|
||
'risked' => ['count' => $result->riskyCount(), 'color' => 'fg-cyan'],
|
||
'incomplete' => ['count' => $result->notImplementedCount(), 'color' => 'fg-cyan'],
|
||
'passed' => ['count' => $this->successfulTestCount($result), 'color' => 'fg-green'],
|
||
];
|
||
|
||
$filteredResults = array_filter($results, function ($item): bool {
|
||
return $item['count'] > 0;
|
||
});
|
||
|
||
foreach ($filteredResults as $key => $info) {
|
||
$this->writeProgressWithColor($info['color'], $info['count']." $key");
|
||
|
||
if ($key !== array_reverse(array_keys($filteredResults))[0]) {
|
||
$this->write(', ');
|
||
}
|
||
}
|
||
|
||
$this->writeNewLine();
|
||
$this->write("Assertions: $this->numAssertions");
|
||
|
||
$this->writeNewLine();
|
||
$this->write("Time: {$result->time()}s");
|
||
|
||
$this->writeNewLine();
|
||
}
|
||
|
||
private function successfulTestCount(TestResult $result): int
|
||
{
|
||
return $result->count()
|
||
- $result->failureCount()
|
||
- $result->errorCount()
|
||
- $result->skippedCount()
|
||
- $result->warningCount()
|
||
- $result->notImplementedCount()
|
||
- $result->riskyCount();
|
||
}
|
||
|
||
/** @phpstan-ignore-next-line */
|
||
public function startTestSuite(TestSuite $suite): void
|
||
{
|
||
if (static::isPestTestSuite($suite)) {
|
||
$this->writeWithColor('fg-white, bold', ' '.substr_replace($suite->getName(), '', 0, 2).' ');
|
||
} elseif ($suite->count() === 1) {
|
||
$this->writeWithColor('fg-white, bold', ' '.$suite->getName());
|
||
}
|
||
|
||
$this->flowId = (int) getmypid();
|
||
|
||
if (!$this->isSummaryTestCountPrinted) {
|
||
$this->printEvent(
|
||
'testCount',
|
||
['count' => $suite->count()]
|
||
);
|
||
$this->isSummaryTestCountPrinted = true;
|
||
}
|
||
|
||
$suiteName = $suite->getName();
|
||
|
||
if (file_exists($suiteName) || !method_exists($suiteName, '__getFileName')) {
|
||
$this->printEvent(
|
||
self::TEST_SUITE_STARTED, [
|
||
self::NAME => $suiteName,
|
||
self::LOCATION_HINT => self::PROTOCOL.$suiteName,
|
||
]);
|
||
|
||
return;
|
||
}
|
||
|
||
$fileName = $suiteName::__getFileName();
|
||
|
||
$this->printEvent(
|
||
self::TEST_SUITE_STARTED, [
|
||
self::NAME => substr($suiteName, 2),
|
||
self::LOCATION_HINT => self::PROTOCOL.$fileName,
|
||
]);
|
||
}
|
||
|
||
/**
|
||
* Verify that the given test suite is a valid Pest suite.
|
||
*
|
||
* @param TestSuite<Test> $suite
|
||
*/
|
||
private static function isPestTestSuite(TestSuite $suite): bool
|
||
{
|
||
return strncmp($suite->getName(), 'P\\', strlen('P\\')) === 0;
|
||
}
|
||
|
||
/**
|
||
* @param array<string, string|int> $params
|
||
*/
|
||
private function printEvent(string $eventName, array $params = []): void
|
||
{
|
||
$this->write("\n##teamcity[{$eventName}");
|
||
|
||
if ($this->flowId !== 0) {
|
||
$params['flowId'] = $this->flowId;
|
||
}
|
||
|
||
foreach ($params as $key => $value) {
|
||
$escapedValue = self::escapeValue((string) $value);
|
||
$this->write(" {$key}='{$escapedValue}'");
|
||
}
|
||
|
||
$this->write("]\n");
|
||
}
|
||
|
||
private static function escapeValue(string $text): string
|
||
{
|
||
return str_replace(
|
||
['|', "'", "\n", "\r", ']', '['],
|
||
['||', "|'", '|n', '|r', '|]', '|['],
|
||
$text
|
||
);
|
||
}
|
||
|
||
/** @phpstan-ignore-next-line */
|
||
public function endTestSuite(TestSuite $suite): void
|
||
{
|
||
$suiteName = $suite->getName();
|
||
|
||
$this->writeNewLine();
|
||
$this->writeNewLine();
|
||
|
||
if (file_exists($suiteName) || !method_exists($suiteName, '__getFileName')) {
|
||
$this->printEvent(
|
||
self::TEST_SUITE_FINISHED, [
|
||
self::NAME => $suiteName,
|
||
self::LOCATION_HINT => self::PROTOCOL.$suiteName,
|
||
]);
|
||
|
||
return;
|
||
}
|
||
|
||
$this->printEvent(
|
||
self::TEST_SUITE_FINISHED, [
|
||
self::NAME => substr($suiteName, 2),
|
||
]);
|
||
}
|
||
|
||
/**
|
||
* @param Test|Testable $test
|
||
*/
|
||
public function startTest(Test $test): void
|
||
{
|
||
if (!TeamCity::isPestTest($test)) {
|
||
$this->phpunitTeamCity->startTest($test);
|
||
|
||
return;
|
||
}
|
||
|
||
$this->printEvent('testStarted', [
|
||
self::NAME => $test->getName(),
|
||
// @phpstan-ignore-next-line
|
||
self::LOCATION_HINT => self::PROTOCOL.$test->toString(),
|
||
]);
|
||
}
|
||
|
||
public static function isPestTest(Test $test): bool
|
||
{
|
||
/** @var array<string, string> $uses */
|
||
$uses = class_uses($test);
|
||
|
||
return in_array(Testable::class, $uses, true);
|
||
}
|
||
|
||
/**
|
||
* @param Test|Testable $test
|
||
*/
|
||
public function endTest(Test $test, float $time): void
|
||
{
|
||
$this->printEvent('testFinished', [
|
||
self::NAME => $test->getName(),
|
||
self::DURATION => self::toMilliseconds($time),
|
||
]);
|
||
|
||
if (!$this->lastTestFailed) {
|
||
$this->writePestTestOutput($test->getName(), 'fg-green, bold', '✓');
|
||
}
|
||
|
||
if ($test instanceof TestCase) {
|
||
$this->numAssertions += $test->getNumAssertions();
|
||
} elseif ($test instanceof PhptTestCase) {
|
||
$this->numAssertions++;
|
||
}
|
||
|
||
$this->lastTestFailed = false;
|
||
}
|
||
|
||
private static function toMilliseconds(float $time): int
|
||
{
|
||
return (int) round($time * 1000);
|
||
}
|
||
|
||
private function writePestTestOutput(string $message, string $color, string $symbol, string $suffix = null): void
|
||
{
|
||
$this->writeProgressWithColor($color, "$symbol ");
|
||
$this->writeProgress($message);
|
||
|
||
if ($suffix !== null && strlen($suffix) > 0) {
|
||
$suffix = str_replace("\n", ' ', $suffix);
|
||
$this->writeWithColor($color, " -> $suffix");
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @param Test|Testable $test
|
||
*/
|
||
public function addError(Test $test, Throwable $t, float $time): void
|
||
{
|
||
$this->lastTestFailed = true;
|
||
$this->writePestTestOutput($test->getName(), 'fg-red, bold', '⨯');
|
||
|
||
$this->phpunitTeamCity->addError($test, $t, $time);
|
||
}
|
||
|
||
public function addFailure(Test $test, AssertionFailedError $e, float $time): void
|
||
{
|
||
$this->lastTestFailed = true;
|
||
$this->writePestTestOutput($test->getName(), 'fg-red, bold', '⨯');
|
||
|
||
$this->phpunitTeamCity->addFailure($test, $e, $time);
|
||
}
|
||
|
||
/**
|
||
* @phpstan-ignore-next-line
|
||
*
|
||
* @param Test|Testable $test
|
||
*/
|
||
public function addWarning(Test $test, Warning $e, float $time): void
|
||
{
|
||
$this->lastTestFailed = true;
|
||
$this->writeWarning($test, $e);
|
||
}
|
||
|
||
private function writeWarning(Test $test, Throwable $t): void
|
||
{
|
||
$this->writePestTestOutput($test->getName(), 'fg-yellow, bold', '-', $t->getMessage());
|
||
}
|
||
|
||
public function addIncompleteTest(Test $test, Throwable $t, float $time): void
|
||
{
|
||
$this->lastTestFailed = true;
|
||
$this->writeWarning($test, $t);
|
||
}
|
||
|
||
public function addRiskyTest(Test $test, Throwable $t, float $time): void
|
||
{
|
||
$this->lastTestFailed = true;
|
||
$this->writeWarning($test, $t);
|
||
}
|
||
|
||
public function addSkippedTest(Test $test, Throwable $t, float $time): void
|
||
{
|
||
$this->lastTestFailed = true;
|
||
$this->writeWarning($test, $t);
|
||
$this->phpunitTeamCity->printIgnoredTest($test->getName(), $t, $time);
|
||
}
|
||
}
|