diff --git a/composer-require-checker.json b/composer-require-checker.json
index 3e87a142..7878977e 100644
--- a/composer-require-checker.json
+++ b/composer-require-checker.json
@@ -8,7 +8,6 @@
"Symfony\\Bridge\\Twig\\TokenParser\\TransTokenParser",
"Symfony\\UX\\TwigComponent\\Twig\\ComponentTokenParser",
"Symfony\\UX\\TwigComponent\\Twig\\PropsTokenParser",
- "Twig\\Extra\\Cache\\TokenParser\\CacheTokenParser",
- "\\SimpleXMLElement"
+ "Twig\\Extra\\Cache\\TokenParser\\CacheTokenParser"
]
}
diff --git a/src/Report/Reporter/CheckstyleReporter.php b/src/Report/Reporter/CheckstyleReporter.php
index 880b8dd8..f37f1219 100644
--- a/src/Report/Reporter/CheckstyleReporter.php
+++ b/src/Report/Reporter/CheckstyleReporter.php
@@ -8,35 +8,52 @@
use TwigCsFixer\Report\Report;
use TwigCsFixer\Report\SniffViolation;
-/**
- * Human-readable output with context.
- */
final class CheckstyleReporter implements ReporterInterface
{
public const NAME = 'checkstyle';
public function display(OutputInterface $output, Report $report, ?string $level = null): void
{
- $checkstyle = new \SimpleXMLElement('');
+ $text = ''."\n";
+
+ $text .= ''."\n";
foreach ($report->getFiles() as $file) {
$fileMessages = $report->getMessages($file, $level);
- if (\count($fileMessages) > 0) {
- /** @var \SimpleXMLElement $fileNode */
- $fileNode = $checkstyle->addChild('file');
- $fileNode->addAttribute('name', $file);
-
- foreach ($fileMessages as $message) {
- /** @var \SimpleXMLElement $violation */
- $violation = $fileNode->addChild('violation');
- $violation->addAttribute('column', (string) $message->getLinePosition());
- $violation->addAttribute('severity', strtolower(SniffViolation::getLevelAsString($message->getLevel())));
- $violation->addAttribute('message', $message->getMessage());
- $violation->addAttribute('source', $message->getSniffName() ?? '');
+ if (0 === \count($fileMessages)) {
+ continue;
+ }
+
+ $text .= ' '."\n";
+ foreach ($fileMessages as $message) {
+ $line = (string) $message->getLine();
+ $linePosition = (string) $message->getLinePosition();
+ $sniffName = $message->getSniffName();
+
+ $text .= ' getLevel())).'"';
+ $text .= ' message="'.$this->xmlEncode($message->getMessage()).'"';
+ if (null !== $sniffName) {
+ $text .= ' source="'.$sniffName.'"';
+ }
+ $text .= '/>'."\n";
}
+ $text .= ' '."\n";
}
- $output->writeln((string) $checkstyle->asXML());
+ $text .= ''."\n";
+
+ $output->writeln($text);
+ }
+
+ private function xmlEncode(string $data): string
+ {
+ return htmlspecialchars($data, \ENT_XML1 | \ENT_QUOTES);
}
}
diff --git a/tests/Command/TwigCsFixerCommandTest.php b/tests/Command/TwigCsFixerCommandTest.php
index 02c3981b..8c151d2b 100644
--- a/tests/Command/TwigCsFixerCommandTest.php
+++ b/tests/Command/TwigCsFixerCommandTest.php
@@ -116,9 +116,9 @@ public function testExecuteWithReportOption(): void
$commandTester = new CommandTester($command);
$commandTester->execute([
- 'paths' => [$this->getTmpPath(__DIR__.'/Fixtures/file.twig')],
+ 'paths' => [$this->getTmpPath(__DIR__.'/Fixtures')],
'--no-cache' => true, // To avoid cache output
- '--report' => 'null',
+ '--report' => 'checkstyle',
]);
static::assertSame('', $commandTester->getDisplay());
diff --git a/tests/Report/Formatter/CheckstyleReporterTest.php b/tests/Report/Formatter/CheckstyleReporterTest.php
new file mode 100644
index 00000000..1198030a
--- /dev/null
+++ b/tests/Report/Formatter/CheckstyleReporterTest.php
@@ -0,0 +1,82 @@
+addMessage($violation0);
+ $violation1 = new SniffViolation(SniffViolation::LEVEL_WARNING, 'Warning', $file, 2, 22, 'WarningSniff');
+ $report->addMessage($violation1);
+ $violation2 = new SniffViolation(SniffViolation::LEVEL_ERROR, 'Error', $file, 3, 33, 'ErrorSniff');
+ $report->addMessage($violation2);
+ $violation3 = new SniffViolation(SniffViolation::LEVEL_FATAL, 'Fatal', $file);
+ $report->addMessage($violation3);
+
+ $output = new BufferedOutput(OutputInterface::VERBOSITY_NORMAL, true);
+ $textFormatter->display($output, $report, $level);
+
+ $text = $output->fetch();
+ static::assertStringContainsString($expected, $text);
+ }
+
+ /**
+ * @return iterable
+ */
+ public static function displayDataProvider(): iterable
+ {
+ yield [
+ sprintf(
+ <<
+
+
+
+
+
+
+
+
+ EOD,
+ __DIR__
+ ),
+ null,
+ ];
+
+ yield [
+ sprintf(
+ <<
+
+
+
+
+
+
+ EOD,
+ __DIR__
+ ),
+ Report::MESSAGE_TYPE_ERROR,
+ ];
+ }
+}