2019-12-19 21:18:09 +01:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Report;
|
|
|
|
|
|
|
|
use DOMDocument;
|
|
|
|
use DOMElement;
|
|
|
|
use Psalm\Config;
|
|
|
|
use Psalm\Report;
|
|
|
|
use function count;
|
|
|
|
use function sprintf;
|
|
|
|
use function trim;
|
|
|
|
use Doctrine\Instantiator\Exception\UnexpectedValueException;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* based on https://github.com/m50/psalm-json-to-junit
|
|
|
|
* Copyright (c) Marisa Clardy marisa@clardy.eu
|
|
|
|
*
|
|
|
|
* with a few modifications
|
|
|
|
*/
|
|
|
|
class JunitReport extends Report
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function create(): string
|
|
|
|
{
|
|
|
|
$errors = 0;
|
|
|
|
$warnings = 0;
|
|
|
|
$tests = 0;
|
|
|
|
|
|
|
|
$ndata = [];
|
|
|
|
|
|
|
|
foreach ($this->issues_data as $error) {
|
2020-02-17 00:24:40 +01:00
|
|
|
$is_error = $error->severity === Config::REPORT_ERROR;
|
|
|
|
$is_warning = $error->severity === Config::REPORT_INFO;
|
2019-12-19 21:18:09 +01:00
|
|
|
|
|
|
|
if ($is_error) {
|
|
|
|
$errors++;
|
|
|
|
} elseif ($is_warning) {
|
|
|
|
$warnings++;
|
|
|
|
} else {
|
|
|
|
// currently this never happens
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$tests++;
|
|
|
|
|
2020-02-17 00:24:40 +01:00
|
|
|
$fname = $error->file_name;
|
2019-12-19 21:18:09 +01:00
|
|
|
|
|
|
|
if (!isset($ndata[$fname])) {
|
|
|
|
$ndata[$fname] = [
|
|
|
|
'errors' => $is_error ? 1 : 0,
|
|
|
|
'warnings' => $is_warning ? 1 : 0,
|
|
|
|
'failures' => [
|
|
|
|
$this->createFailure($error),
|
|
|
|
],
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
if ($is_error) {
|
|
|
|
$ndata[$fname]['errors']++;
|
|
|
|
} else {
|
|
|
|
$ndata[$fname]['warnings']++;
|
|
|
|
}
|
|
|
|
|
|
|
|
$ndata[$fname]['failures'][] = $this->createFailure($error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$dom = new DOMDocument('1.0', 'UTF-8');
|
|
|
|
$dom->formatOutput = true;
|
|
|
|
|
|
|
|
$schema = 'https://raw.githubusercontent.com/junit-team/'.
|
|
|
|
'junit5/r5.5.1/platform-tests/src/test/resources/jenkins-junit.xsd';
|
|
|
|
|
|
|
|
$suites = $dom->createElement('testsuites');
|
|
|
|
|
2020-03-05 05:05:11 +01:00
|
|
|
$suites->setAttribute('failures', (string) $errors);
|
|
|
|
$suites->setAttribute('errors', "0");
|
|
|
|
$suites->setAttribute('name', 'psalm');
|
|
|
|
$suites->setAttribute('tests', (string) $tests);
|
|
|
|
$suites->setAttribute('xmlns:xsi', 'http://www.w3.org/2001/XMLSchema-instance');
|
|
|
|
$suites->setAttribute('xsi:noNamespaceSchemaLocation', $schema);
|
2019-12-19 21:18:09 +01:00
|
|
|
$dom->appendChild($suites);
|
|
|
|
|
|
|
|
if (!count($ndata)) {
|
|
|
|
$testcase = $dom->createElement('testcase');
|
|
|
|
$testcase->setAttribute('name', 'psalm');
|
2020-03-05 05:05:11 +01:00
|
|
|
$suites->appendChild($testcase);
|
2019-12-19 21:18:09 +01:00
|
|
|
} else {
|
|
|
|
foreach ($ndata as $file => $report) {
|
2020-03-05 05:05:11 +01:00
|
|
|
$this->createTestSuite($dom, $suites, $file, $report);
|
2019-12-19 21:18:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return $dom->saveXML();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array{
|
|
|
|
* errors: int,
|
|
|
|
* warnings: int,
|
|
|
|
* failures: list<array{
|
|
|
|
* data: array{
|
|
|
|
* column_from: int,
|
|
|
|
* column_to: int,
|
|
|
|
* line: int,
|
|
|
|
* message: string,
|
|
|
|
* selected_text: string,
|
|
|
|
* snippet: string,
|
|
|
|
* type: string},
|
|
|
|
* type: string
|
|
|
|
* }>
|
|
|
|
* } $report
|
|
|
|
*/
|
|
|
|
private function createTestSuite(DOMDocument $dom, DOMElement $parent, string $file, array $report): void
|
|
|
|
{
|
|
|
|
$totalTests = $report['errors'] + $report['warnings'];
|
|
|
|
if ($totalTests < 1) {
|
|
|
|
$totalTests = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
$testsuite = $dom->createElement('testsuite');
|
|
|
|
$testsuite->setAttribute('name', $file);
|
|
|
|
$testsuite->setAttribute('file', $file);
|
|
|
|
$testsuite->setAttribute('assertions', (string) $totalTests);
|
|
|
|
$testsuite->setAttribute('failures', (string) $report['errors']);
|
|
|
|
$testsuite->setAttribute('warnings', (string) $report['warnings']);
|
|
|
|
|
|
|
|
$failuresByType = $this->groupByType($report['failures']);
|
|
|
|
$testsuite->setAttribute('tests', (string) count($failuresByType));
|
|
|
|
|
|
|
|
$iterator = 0;
|
|
|
|
foreach ($failuresByType as $type => $data) {
|
|
|
|
foreach ($data as $d) {
|
|
|
|
$testcase = $dom->createElement('testcase');
|
|
|
|
$testcase->setAttribute('name', "{$file}:{$d['line']}");
|
|
|
|
$testcase->setAttribute('file', $file);
|
|
|
|
$testcase->setAttribute('class', $type);
|
|
|
|
$testcase->setAttribute('classname', $type);
|
|
|
|
$testcase->setAttribute('line', (string) $d['line']);
|
|
|
|
$testcase->setAttribute('assertions', (string) count($data));
|
|
|
|
|
|
|
|
$failure = $dom->createElement('failure');
|
|
|
|
$failure->setAttribute('type', $type);
|
|
|
|
$failure->nodeValue = $this->dataToOutput($d);
|
|
|
|
|
|
|
|
$testcase->appendChild($failure);
|
|
|
|
$testsuite->appendChild($testcase);
|
|
|
|
}
|
|
|
|
$iterator++;
|
|
|
|
}
|
|
|
|
$parent->appendChild($testsuite);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array{
|
|
|
|
* data: array{
|
|
|
|
* column_from: int,
|
|
|
|
* column_to: int,
|
|
|
|
* line: int,
|
|
|
|
* message: string,
|
|
|
|
* selected_text: string,
|
|
|
|
* snippet: string,
|
|
|
|
* type: string
|
|
|
|
* },
|
|
|
|
* type: string
|
|
|
|
* }
|
|
|
|
*/
|
2020-02-17 00:24:40 +01:00
|
|
|
private function createFailure(\Psalm\Internal\Analyzer\IssueData $issue_data) : array
|
2019-12-19 21:18:09 +01:00
|
|
|
{
|
|
|
|
return [
|
2020-02-17 00:24:40 +01:00
|
|
|
'type' => $issue_data->type,
|
2019-12-19 21:18:09 +01:00
|
|
|
'data' => [
|
2020-02-17 00:24:40 +01:00
|
|
|
'message' => $issue_data->message,
|
|
|
|
'type' => $issue_data->type,
|
|
|
|
'snippet' => $issue_data->snippet,
|
|
|
|
'selected_text' => $issue_data->selected_text,
|
|
|
|
'line' => $issue_data->line_from,
|
|
|
|
'column_from' => $issue_data->column_from,
|
|
|
|
'column_to' => $issue_data->column_to,
|
2019-12-19 21:18:09 +01:00
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array<array{
|
|
|
|
* data: array{
|
|
|
|
* column_from: int,
|
|
|
|
* column_to: int,
|
|
|
|
* line: int,
|
|
|
|
* message: string,
|
|
|
|
* selected_text: string,
|
|
|
|
* snippet: string,
|
|
|
|
* type: string
|
|
|
|
* },
|
|
|
|
* type: string
|
|
|
|
* }> $failures
|
|
|
|
*
|
|
|
|
* @return array<string, non-empty-list<array{
|
|
|
|
* column_from: int,
|
|
|
|
* column_to: int,
|
|
|
|
* line: int,
|
|
|
|
* message: string,
|
|
|
|
* selected_text: string,
|
|
|
|
* snippet: string,
|
|
|
|
* type: string
|
|
|
|
* }>>
|
|
|
|
*/
|
|
|
|
private function groupByType(array $failures)
|
|
|
|
{
|
|
|
|
$nfailures = [];
|
|
|
|
|
|
|
|
foreach ($failures as $failure) {
|
|
|
|
$nfailures[$failure['type']][] = $failure['data'];
|
|
|
|
}
|
|
|
|
|
|
|
|
return $nfailures;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array<string, int|string> $data
|
|
|
|
*/
|
|
|
|
private function dataToOutput(array $data): string
|
|
|
|
{
|
|
|
|
$ret = '';
|
|
|
|
|
|
|
|
foreach ($data as $key => $value) {
|
|
|
|
$value = trim((string) $value);
|
|
|
|
$ret .= "{$key}: {$value}\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
}
|