mirror of
https://github.com/pestphp/pest.git
synced 2026-03-06 07:47:22 +01:00
385 lines
12 KiB
PHP
385 lines
12 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Pest\Logging\JUnit;
|
|
|
|
use DOMDocument;
|
|
use DOMElement;
|
|
use Pest\Logging\Converter;
|
|
use Pest\Logging\JUnit\Subscriber\TestErroredSubscriber;
|
|
use Pest\Logging\JUnit\Subscriber\TestFailedSubscriber;
|
|
use Pest\Logging\JUnit\Subscriber\TestFinishedSubscriber;
|
|
use Pest\Logging\JUnit\Subscriber\TestMarkedIncompleteSubscriber;
|
|
use Pest\Logging\JUnit\Subscriber\TestPreparedSubscriber;
|
|
use Pest\Logging\JUnit\Subscriber\TestRunnerExecutionFinishedSubscriber;
|
|
use Pest\Logging\JUnit\Subscriber\TestSkippedSubscriber;
|
|
use Pest\Logging\JUnit\Subscriber\TestSuiteFinishedSubscriber;
|
|
use Pest\Logging\JUnit\Subscriber\TestSuiteStartedSubscriber;
|
|
use PHPUnit\Event\EventFacadeIsSealedException;
|
|
use PHPUnit\Event\Facade;
|
|
use PHPUnit\Event\InvalidArgumentException;
|
|
use PHPUnit\Event\Telemetry\HRTime;
|
|
use PHPUnit\Event\Telemetry\Info;
|
|
use PHPUnit\Event\Test\Errored;
|
|
use PHPUnit\Event\Test\Failed;
|
|
use PHPUnit\Event\Test\Finished;
|
|
use PHPUnit\Event\Test\MarkedIncomplete;
|
|
use PHPUnit\Event\Test\Prepared;
|
|
use PHPUnit\Event\Test\Skipped;
|
|
use PHPUnit\Event\TestData\NoDataSetFromDataProviderException;
|
|
use PHPUnit\Event\TestSuite\Started;
|
|
use PHPUnit\Event\UnknownSubscriberTypeException;
|
|
use PHPUnit\TextUI\Output\Printer;
|
|
use PHPUnit\Util\Xml;
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
final class JUnitLogger
|
|
{
|
|
private DOMDocument $document;
|
|
|
|
private DOMElement $root;
|
|
|
|
/**
|
|
* @var DOMElement[]
|
|
*/
|
|
private array $testSuites = [];
|
|
|
|
/**
|
|
* @psalm-var array<int,int>
|
|
*/
|
|
private array $testSuiteTests = [0];
|
|
|
|
/**
|
|
* @psalm-var array<int,int>
|
|
*/
|
|
private array $testSuiteAssertions = [0];
|
|
|
|
/**
|
|
* @psalm-var array<int,int>
|
|
*/
|
|
private array $testSuiteErrors = [0];
|
|
|
|
/**
|
|
* @psalm-var array<int,int>
|
|
*/
|
|
private array $testSuiteFailures = [0];
|
|
|
|
/**
|
|
* @psalm-var array<int,int>
|
|
*/
|
|
private array $testSuiteSkipped = [0];
|
|
|
|
/**
|
|
* @psalm-var array<int,int|float>
|
|
*/
|
|
private array $testSuiteTimes = [0];
|
|
|
|
private int $testSuiteLevel = 0;
|
|
|
|
private ?DOMElement $currentTestCase = null;
|
|
|
|
private ?HRTime $time = null;
|
|
|
|
private bool $prepared = false;
|
|
|
|
/**
|
|
* @throws EventFacadeIsSealedException
|
|
* @throws UnknownSubscriberTypeException
|
|
*/
|
|
public function __construct(
|
|
private readonly Printer $printer,
|
|
private readonly Converter $converter,
|
|
) {
|
|
$this->registerSubscribers();
|
|
$this->createDocument();
|
|
}
|
|
|
|
public function flush(): void
|
|
{
|
|
$this->printer->print((string) $this->document->saveXML());
|
|
|
|
$this->printer->flush();
|
|
}
|
|
|
|
public function testSuiteStarted(Started $event): void
|
|
{
|
|
$testSuite = $this->document->createElement('testsuite');
|
|
$testSuite->setAttribute('name', $this->converter->getTestSuiteName($event->testSuite()));
|
|
|
|
if ($event->testSuite()->isForTestClass()) {
|
|
$testSuite->setAttribute('file', $this->converter->getTestSuiteLocation($event->testSuite()) ?? '');
|
|
}
|
|
|
|
if ($this->testSuiteLevel > 0) {
|
|
$this->testSuites[$this->testSuiteLevel]->appendChild($testSuite);
|
|
} else {
|
|
$this->root->appendChild($testSuite);
|
|
}
|
|
|
|
$this->testSuiteLevel++;
|
|
$this->testSuites[$this->testSuiteLevel] = $testSuite;
|
|
$this->testSuiteTests[$this->testSuiteLevel] = 0;
|
|
$this->testSuiteAssertions[$this->testSuiteLevel] = 0;
|
|
$this->testSuiteErrors[$this->testSuiteLevel] = 0;
|
|
$this->testSuiteFailures[$this->testSuiteLevel] = 0;
|
|
$this->testSuiteSkipped[$this->testSuiteLevel] = 0;
|
|
$this->testSuiteTimes[$this->testSuiteLevel] = 0;
|
|
}
|
|
|
|
public function testSuiteFinished(): void
|
|
{
|
|
$this->testSuites[$this->testSuiteLevel]->setAttribute(
|
|
'tests',
|
|
(string) $this->testSuiteTests[$this->testSuiteLevel],
|
|
);
|
|
|
|
$this->testSuites[$this->testSuiteLevel]->setAttribute(
|
|
'assertions',
|
|
(string) $this->testSuiteAssertions[$this->testSuiteLevel],
|
|
);
|
|
|
|
$this->testSuites[$this->testSuiteLevel]->setAttribute(
|
|
'errors',
|
|
(string) $this->testSuiteErrors[$this->testSuiteLevel],
|
|
);
|
|
|
|
$this->testSuites[$this->testSuiteLevel]->setAttribute(
|
|
'failures',
|
|
(string) $this->testSuiteFailures[$this->testSuiteLevel],
|
|
);
|
|
|
|
$this->testSuites[$this->testSuiteLevel]->setAttribute(
|
|
'skipped',
|
|
(string) $this->testSuiteSkipped[$this->testSuiteLevel],
|
|
);
|
|
|
|
$this->testSuites[$this->testSuiteLevel]->setAttribute(
|
|
'time',
|
|
sprintf('%F', $this->testSuiteTimes[$this->testSuiteLevel]),
|
|
);
|
|
|
|
if ($this->testSuiteLevel > 1) {
|
|
$this->testSuiteTests[$this->testSuiteLevel - 1] += $this->testSuiteTests[$this->testSuiteLevel];
|
|
$this->testSuiteAssertions[$this->testSuiteLevel - 1] += $this->testSuiteAssertions[$this->testSuiteLevel];
|
|
$this->testSuiteErrors[$this->testSuiteLevel - 1] += $this->testSuiteErrors[$this->testSuiteLevel];
|
|
$this->testSuiteFailures[$this->testSuiteLevel - 1] += $this->testSuiteFailures[$this->testSuiteLevel];
|
|
$this->testSuiteSkipped[$this->testSuiteLevel - 1] += $this->testSuiteSkipped[$this->testSuiteLevel];
|
|
$this->testSuiteTimes[$this->testSuiteLevel - 1] += $this->testSuiteTimes[$this->testSuiteLevel];
|
|
}
|
|
|
|
$this->testSuiteLevel--;
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
* @throws NoDataSetFromDataProviderException
|
|
*/
|
|
public function testPrepared(Prepared $event): void
|
|
{
|
|
$this->createTestCase($event);
|
|
$this->prepared = true;
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
*/
|
|
public function testFinished(Finished $event): void
|
|
{
|
|
$this->handleFinish($event->telemetryInfo(), $event->numberOfAssertionsPerformed());
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
* @throws NoDataSetFromDataProviderException
|
|
*/
|
|
public function testMarkedIncomplete(MarkedIncomplete $event): void
|
|
{
|
|
$this->handleIncompleteOrSkipped($event);
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
* @throws NoDataSetFromDataProviderException
|
|
*/
|
|
public function testSkipped(Skipped $event): void
|
|
{
|
|
$this->handleIncompleteOrSkipped($event);
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
* @throws NoDataSetFromDataProviderException
|
|
*/
|
|
public function testErrored(Errored $event): void
|
|
{
|
|
$this->handleFault($event, 'error');
|
|
|
|
$this->testSuiteErrors[$this->testSuiteLevel]++;
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
* @throws NoDataSetFromDataProviderException
|
|
*/
|
|
public function testFailed(Failed $event): void
|
|
{
|
|
$this->handleFault($event, 'failure');
|
|
|
|
$this->testSuiteFailures[$this->testSuiteLevel]++;
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
*/
|
|
private function handleFinish(Info $telemetryInfo, int $numberOfAssertionsPerformed): void
|
|
{
|
|
assert($this->currentTestCase instanceof \DOMElement);
|
|
assert($this->time instanceof \PHPUnit\Event\Telemetry\HRTime);
|
|
|
|
$time = $telemetryInfo->time()->duration($this->time)->asFloat();
|
|
|
|
$this->testSuiteAssertions[$this->testSuiteLevel] += $numberOfAssertionsPerformed;
|
|
|
|
$this->currentTestCase->setAttribute(
|
|
'assertions',
|
|
(string) $numberOfAssertionsPerformed,
|
|
);
|
|
|
|
$this->currentTestCase->setAttribute(
|
|
'time',
|
|
sprintf('%F', $time),
|
|
);
|
|
|
|
$this->testSuites[$this->testSuiteLevel]->appendChild(
|
|
$this->currentTestCase,
|
|
);
|
|
|
|
$this->testSuiteTests[$this->testSuiteLevel]++;
|
|
$this->testSuiteTimes[$this->testSuiteLevel] += $time;
|
|
|
|
$this->currentTestCase = null;
|
|
$this->time = null;
|
|
$this->prepared = false;
|
|
}
|
|
|
|
/**
|
|
* @throws EventFacadeIsSealedException
|
|
* @throws UnknownSubscriberTypeException
|
|
*/
|
|
private function registerSubscribers(): void
|
|
{
|
|
$subscribers = [
|
|
new TestSuiteStartedSubscriber($this),
|
|
new TestSuiteFinishedSubscriber($this),
|
|
new TestPreparedSubscriber($this),
|
|
new TestFinishedSubscriber($this),
|
|
new TestErroredSubscriber($this),
|
|
new TestFailedSubscriber($this),
|
|
new TestMarkedIncompleteSubscriber($this),
|
|
new TestSkippedSubscriber($this),
|
|
new TestRunnerExecutionFinishedSubscriber($this),
|
|
];
|
|
|
|
Facade::instance()->registerSubscribers(...$subscribers);
|
|
}
|
|
|
|
private function createDocument(): void
|
|
{
|
|
$this->document = new DOMDocument('1.0', 'UTF-8');
|
|
$this->document->formatOutput = true;
|
|
|
|
$this->root = $this->document->createElement('testsuites');
|
|
$this->document->appendChild($this->root);
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
* @throws NoDataSetFromDataProviderException
|
|
*/
|
|
private function handleFault(Errored|Failed $event, string $type): void
|
|
{
|
|
if (! $this->prepared) {
|
|
$this->createTestCase($event);
|
|
}
|
|
|
|
assert($this->currentTestCase instanceof \DOMElement);
|
|
|
|
$throwable = $event->throwable();
|
|
|
|
$testName = $this->converter->getTestCaseMethodName($event->test());
|
|
$message = $this->converter->getExceptionMessage($throwable);
|
|
$details = $this->converter->getExceptionDetails($throwable);
|
|
|
|
$buffer = $testName;
|
|
$buffer .= trim(
|
|
$message.PHP_EOL.
|
|
$details,
|
|
);
|
|
|
|
$fault = $this->document->createElement(
|
|
$type,
|
|
Xml::prepareString($buffer),
|
|
);
|
|
|
|
$fault->setAttribute('type', $throwable->className());
|
|
|
|
$this->currentTestCase->appendChild($fault);
|
|
|
|
if (! $this->prepared) {
|
|
$this->handleFinish($event->telemetryInfo(), 0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
* @throws NoDataSetFromDataProviderException
|
|
*/
|
|
private function handleIncompleteOrSkipped(MarkedIncomplete|Skipped $event): void
|
|
{
|
|
if (! $this->prepared) {
|
|
$this->createTestCase($event);
|
|
}
|
|
|
|
assert($this->currentTestCase instanceof \DOMElement);
|
|
|
|
$skipped = $this->document->createElement('skipped');
|
|
|
|
$this->currentTestCase->appendChild($skipped);
|
|
|
|
$this->testSuiteSkipped[$this->testSuiteLevel]++;
|
|
|
|
if (! $this->prepared) {
|
|
$this->handleFinish($event->telemetryInfo(), 0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @throws InvalidArgumentException
|
|
* @throws NoDataSetFromDataProviderException
|
|
*
|
|
* @psalm-assert !null $this->currentTestCase
|
|
*/
|
|
private function createTestCase(Errored|Failed|MarkedIncomplete|Prepared|Skipped $event): void
|
|
{
|
|
$testCase = $this->document->createElement('testcase');
|
|
|
|
$test = $event->test();
|
|
$file = $this->converter->getTestCaseLocation($test);
|
|
|
|
$testCase->setAttribute('name', $this->converter->getTestCaseMethodName($test));
|
|
$testCase->setAttribute('file', $file);
|
|
|
|
if ($test->isTestMethod()) {
|
|
$className = $this->converter->getTrimmedTestClassName($test);
|
|
$testCase->setAttribute('class', $className);
|
|
$testCase->setAttribute('classname', str_replace('\\', '.', $className));
|
|
}
|
|
|
|
$this->currentTestCase = $testCase;
|
|
$this->time = $event->telemetryInfo()->time();
|
|
}
|
|
}
|