2020-11-23 21:34:51 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Psalm\Report;
|
|
|
|
|
|
|
|
use Psalm\Config;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Internal\Analyzer\IssueData;
|
2020-11-23 21:34:51 +01:00
|
|
|
use Psalm\Internal\Json\Json;
|
|
|
|
use Psalm\Report;
|
|
|
|
|
2021-12-03 21:07:25 +01:00
|
|
|
use function array_map;
|
2020-11-23 21:34:51 +01:00
|
|
|
use function array_values;
|
|
|
|
use function md5;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CodeClimate format
|
|
|
|
* This is the format used by Gitlab for CodeQuality
|
|
|
|
*
|
|
|
|
* @see https://docs.gitlab.com/ee/user/project/merge_requests/code_quality.html
|
|
|
|
* @see https://github.com/codeclimate/platform/blob/master/spec/analyzers/SPEC.md#data-types
|
|
|
|
*
|
|
|
|
* @author Olivier Doucet <webmaster@ajeux.com>
|
|
|
|
*/
|
|
|
|
class CodeClimateReport extends Report
|
|
|
|
{
|
|
|
|
public function create(): string
|
|
|
|
{
|
|
|
|
$options = $this->pretty ? Json::PRETTY : Json::DEFAULT;
|
|
|
|
|
2021-12-03 21:07:25 +01:00
|
|
|
$issues_data = array_map(
|
2022-01-05 23:45:11 +01:00
|
|
|
fn(IssueData $issue): array => [
|
|
|
|
'type' => 'issue',
|
|
|
|
'check_name' => $issue->type,
|
|
|
|
'description' => $issue->message,
|
|
|
|
'categories' => [$issue->type],
|
|
|
|
'severity' => $this->convertSeverity($issue->severity),
|
|
|
|
'fingerprint' => $this->calculateFingerprint($issue),
|
|
|
|
'location' => [
|
|
|
|
'path' => $issue->file_name,
|
|
|
|
'lines' => [
|
|
|
|
'begin' => $issue->line_from,
|
|
|
|
'end' => $issue->line_to,
|
2020-11-23 21:34:51 +01:00
|
|
|
],
|
2022-01-05 23:45:11 +01:00
|
|
|
],
|
|
|
|
],
|
2020-11-23 21:34:51 +01:00
|
|
|
$this->issues_data
|
|
|
|
);
|
|
|
|
|
|
|
|
return Json::encode(array_values($issues_data), $options) . "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* convert our own severity to CodeClimate format
|
|
|
|
* Values can be : info, minor, major, critical, or blocker
|
|
|
|
*/
|
|
|
|
protected function convertSeverity(string $input): string
|
|
|
|
{
|
|
|
|
if (Config::REPORT_INFO === $input) {
|
|
|
|
return 'info';
|
|
|
|
}
|
|
|
|
if (Config::REPORT_ERROR === $input) {
|
|
|
|
return 'critical';
|
|
|
|
}
|
|
|
|
if (Config::REPORT_SUPPRESS === $input) {
|
|
|
|
return 'minor';
|
|
|
|
}
|
|
|
|
|
|
|
|
// unknown cases ? fallback
|
|
|
|
return 'critical';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calculate a unique fingerprint for a given issue
|
|
|
|
*/
|
|
|
|
protected function calculateFingerprint(IssueData $issue): string
|
|
|
|
{
|
2021-07-16 22:35:16 +02:00
|
|
|
return md5($issue->type.$issue->message.$issue->file_name.$issue->from.$issue->to);
|
2020-11-23 21:34:51 +01:00
|
|
|
}
|
|
|
|
}
|