2016-06-05 20:25:16 -04:00
|
|
|
<?php
|
2016-07-25 18:37:44 -04:00
|
|
|
namespace Psalm;
|
2016-06-05 20:25:16 -04:00
|
|
|
|
2017-09-08 17:18:48 +02:00
|
|
|
use LSS\Array2XML;
|
2018-11-05 21:57:36 -05:00
|
|
|
use Psalm\Internal\Analyzer\ProjectAnalyzer;
|
2018-03-20 22:36:03 -04:00
|
|
|
use Psalm\Issue\ClassIssue;
|
2017-03-13 18:06:56 -04:00
|
|
|
use Psalm\Issue\CodeIssue;
|
2018-03-20 22:36:03 -04:00
|
|
|
use Psalm\Issue\MethodIssue;
|
2018-05-11 00:07:41 -04:00
|
|
|
use Psalm\Issue\PropertyIssue;
|
2016-08-13 14:20:46 -04:00
|
|
|
|
2016-06-26 15:18:40 -04:00
|
|
|
class IssueBuffer
|
2016-06-05 20:25:16 -04:00
|
|
|
{
|
2017-07-25 16:11:02 -04:00
|
|
|
/**
|
2018-02-19 18:16:09 -05:00
|
|
|
* @var array<int, array{severity: string, line_from: int, line_to: int, type: string, message: string,
|
|
|
|
* file_name: string, file_path: string, snippet: string, from: int, to: int,
|
2018-11-09 00:46:13 -05:00
|
|
|
* snippet_from: int, snippet_to: int, column_from: int, column_to: int, selected_text: string}>
|
2017-07-25 16:11:02 -04:00
|
|
|
*/
|
|
|
|
protected static $issues_data = [];
|
|
|
|
|
2016-12-07 22:38:57 -05:00
|
|
|
/**
|
|
|
|
* @var array<int, array>
|
|
|
|
*/
|
2017-07-25 16:11:02 -04:00
|
|
|
protected static $console_issues = [];
|
2016-12-07 22:38:57 -05:00
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2017-01-14 01:24:27 -05:00
|
|
|
* @var int
|
2016-11-02 02:29:00 -04:00
|
|
|
*/
|
2017-01-14 01:24:27 -05:00
|
|
|
protected static $error_count = 0;
|
2016-11-01 00:39:41 -04:00
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
|
|
|
* @var array<string, bool>
|
|
|
|
*/
|
2016-10-18 18:55:53 -04:00
|
|
|
protected static $emitted = [];
|
2016-06-20 19:30:38 -04:00
|
|
|
|
2017-03-13 18:06:56 -04:00
|
|
|
/** @var int */
|
|
|
|
protected static $recording_level = 0;
|
|
|
|
|
|
|
|
/** @var array<int, array<int, CodeIssue>> */
|
|
|
|
protected static $recorded_issues = [];
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2017-03-13 18:06:56 -04:00
|
|
|
* @param CodeIssue $e
|
|
|
|
* @param array $suppressed_issues
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2016-11-02 02:29:00 -04:00
|
|
|
* @return bool
|
|
|
|
*/
|
2017-03-13 18:06:56 -04:00
|
|
|
public static function accepts(CodeIssue $e, array $suppressed_issues = [])
|
2016-06-05 20:25:16 -04:00
|
|
|
{
|
2016-06-09 18:08:25 -04:00
|
|
|
$config = Config::getInstance();
|
|
|
|
|
2016-07-26 15:00:40 -04:00
|
|
|
$fqcn_parts = explode('\\', get_class($e));
|
|
|
|
$issue_type = array_pop($fqcn_parts);
|
2016-07-22 13:29:46 -04:00
|
|
|
|
2017-05-26 20:05:57 -04:00
|
|
|
if (in_array($issue_type, $suppressed_issues, true)) {
|
2016-07-22 13:29:46 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-03-24 18:34:46 -04:00
|
|
|
if (!$config->reportIssueInFile($issue_type, $e->getFilePath())) {
|
2016-06-09 18:08:25 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-20 22:36:03 -04:00
|
|
|
if ($e instanceof ClassIssue
|
|
|
|
&& $config->getReportingLevelForClass($issue_type, $e->fq_classlike_name) === Config::REPORT_SUPPRESS
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($e instanceof MethodIssue
|
|
|
|
&& $config->getReportingLevelForMethod($issue_type, $e->method_id) === Config::REPORT_SUPPRESS
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-05-11 00:07:41 -04:00
|
|
|
if ($e instanceof PropertyIssue
|
|
|
|
&& $config->getReportingLevelForProperty($issue_type, $e->property_id) === Config::REPORT_SUPPRESS
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-18 17:26:28 -04:00
|
|
|
$parent_issue_type = self::getParentIssueType($issue_type);
|
2018-03-18 16:39:34 -04:00
|
|
|
|
2018-03-18 17:26:28 -04:00
|
|
|
if ($parent_issue_type) {
|
|
|
|
if (in_array($parent_issue_type, $suppressed_issues, true)) {
|
2018-03-18 16:39:34 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-18 17:26:28 -04:00
|
|
|
if (!$config->reportIssueInFile($parent_issue_type, $e->getFilePath())) {
|
2018-03-18 16:39:34 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-13 18:06:56 -04:00
|
|
|
if (self::$recording_level > 0) {
|
|
|
|
self::$recorded_issues[self::$recording_level][] = $e;
|
2017-05-24 22:07:49 -04:00
|
|
|
|
2017-03-13 18:06:56 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-08-08 14:36:18 -04:00
|
|
|
return self::add($e);
|
2016-06-27 13:22:16 -04:00
|
|
|
}
|
|
|
|
|
2018-03-18 17:26:28 -04:00
|
|
|
/**
|
|
|
|
* @param string $issue_type
|
|
|
|
* @return string|null
|
|
|
|
*/
|
|
|
|
private static function getParentIssueType($issue_type)
|
|
|
|
{
|
|
|
|
if (strpos($issue_type, 'Possibly') === 0) {
|
|
|
|
$stripped_issue_type = preg_replace('/^Possibly(False|Null)?/', '', $issue_type);
|
|
|
|
|
|
|
|
if (strpos($stripped_issue_type, 'Invalid') === false && strpos($stripped_issue_type, 'Un') !== 0) {
|
|
|
|
$stripped_issue_type = 'Invalid' . $stripped_issue_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $stripped_issue_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (preg_match('/^(False|Null)[A-Z]/', $issue_type)) {
|
|
|
|
return preg_replace('/^(False|Null)/', 'Invalid', $issue_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2017-03-13 18:06:56 -04:00
|
|
|
* @param CodeIssue $e
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2016-11-02 02:29:00 -04:00
|
|
|
* @throws Exception\CodeException
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
|
|
|
* @return bool
|
2016-11-02 02:29:00 -04:00
|
|
|
*/
|
2017-03-13 18:06:56 -04:00
|
|
|
public static function add(CodeIssue $e)
|
2016-06-27 13:22:16 -04:00
|
|
|
{
|
|
|
|
$config = Config::getInstance();
|
|
|
|
|
2016-07-26 15:00:40 -04:00
|
|
|
$fqcn_parts = explode('\\', get_class($e));
|
|
|
|
$issue_type = array_pop($fqcn_parts);
|
2016-06-17 17:34:52 -04:00
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
$project_analyzer = ProjectAnalyzer::getInstance();
|
2018-01-05 19:49:27 -05:00
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
if (!$project_analyzer->show_issues) {
|
2018-01-05 19:49:27 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-12-07 22:38:57 -05:00
|
|
|
$error_message = $issue_type . ' - ' . $e->getShortLocation() . ' - ' . $e->getMessage();
|
2016-07-22 13:29:46 -04:00
|
|
|
|
2017-02-12 13:38:41 -05:00
|
|
|
$reporting_level = $config->getReportingLevelForFile($issue_type, $e->getFilePath());
|
2016-06-26 22:03:37 -04:00
|
|
|
|
2018-03-18 17:26:28 -04:00
|
|
|
$parent_issue_type = self::getParentIssueType($issue_type);
|
|
|
|
|
|
|
|
if ($parent_issue_type && $reporting_level === Config::REPORT_ERROR) {
|
|
|
|
$parent_reporting_level = $config->getReportingLevelForFile($parent_issue_type, $e->getFilePath());
|
|
|
|
|
|
|
|
if ($parent_reporting_level !== $reporting_level) {
|
|
|
|
$reporting_level = $parent_reporting_level;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-03 19:11:30 -05:00
|
|
|
if ($reporting_level === Config::REPORT_SUPPRESS) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($reporting_level === Config::REPORT_INFO) {
|
2018-09-26 18:08:23 -04:00
|
|
|
if (!self::alreadyEmitted($error_message)) {
|
2017-07-25 16:11:02 -04:00
|
|
|
self::$issues_data[] = $e->toArray(Config::REPORT_INFO);
|
2016-12-03 19:11:30 -05:00
|
|
|
}
|
2017-05-24 22:07:49 -04:00
|
|
|
|
2016-12-03 19:11:30 -05:00
|
|
|
return false;
|
2016-06-10 14:47:44 -04:00
|
|
|
}
|
|
|
|
|
2016-06-26 22:40:57 -04:00
|
|
|
if ($config->throw_exception) {
|
|
|
|
throw new Exception\CodeException($error_message);
|
|
|
|
}
|
|
|
|
|
2016-10-18 18:55:53 -04:00
|
|
|
if (!self::alreadyEmitted($error_message)) {
|
2018-09-25 18:37:24 -04:00
|
|
|
++self::$error_count;
|
2017-07-25 16:11:02 -04:00
|
|
|
self::$issues_data[] = $e->toArray(Config::REPORT_ERROR);
|
2016-10-18 18:55:53 -04:00
|
|
|
}
|
2016-06-16 19:02:29 -04:00
|
|
|
|
2016-08-08 14:36:18 -04:00
|
|
|
return true;
|
2016-06-05 20:25:16 -04:00
|
|
|
}
|
2016-06-20 19:30:38 -04:00
|
|
|
|
2016-12-07 22:38:57 -05:00
|
|
|
/**
|
2018-02-19 18:16:09 -05:00
|
|
|
* @param array{severity: string, line_from: int, line_to: int, type: string, message: string,
|
|
|
|
* file_name: string, file_path: string, snippet: string, from: int, to: int,
|
|
|
|
* snippet_from: int, snippet_to: int, column_from: int, column_to: int} $issue_data
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2017-07-25 16:11:02 -04:00
|
|
|
* @return string
|
2016-12-07 22:38:57 -05:00
|
|
|
*/
|
2017-07-25 16:11:02 -04:00
|
|
|
protected static function getEmacsOutput(array $issue_data)
|
2016-12-07 22:38:57 -05:00
|
|
|
{
|
2018-02-19 18:16:09 -05:00
|
|
|
return $issue_data['file_path'] . ':' . $issue_data['line_from'] . ':' . $issue_data['column_from'] . ':' .
|
2017-07-25 16:11:02 -04:00
|
|
|
($issue_data['severity'] === Config::REPORT_ERROR ? 'error' : 'warning') . ' - ' . $issue_data['message'];
|
2016-12-07 22:38:57 -05:00
|
|
|
}
|
|
|
|
|
2017-01-15 22:39:26 -05:00
|
|
|
/**
|
2018-02-18 21:57:06 -08:00
|
|
|
* @param array{severity: string, line_from: int, line_to: int, type: string, message: string,
|
|
|
|
* file_name: string, file_path: string, snippet: string, from: int, to: int,
|
|
|
|
* snippet_from: int, snippet_to: int, column_from: int, column_to: int} $issue_data
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected static function getPylintOutput(array $issue_data)
|
|
|
|
{
|
|
|
|
$message = sprintf(
|
|
|
|
'%s: %s',
|
|
|
|
$issue_data['type'],
|
|
|
|
$issue_data['message']
|
|
|
|
);
|
|
|
|
if ($issue_data['severity'] === Config::REPORT_ERROR) {
|
|
|
|
$code = 'E0001';
|
|
|
|
} else {
|
|
|
|
$code = 'W0001';
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://docs.pylint.org/en/1.6.0/output.html doesn't mention what to do about 'column',
|
|
|
|
// but it's still useful for users.
|
|
|
|
// E.g. jenkins can't parse %s:%d:%d.
|
|
|
|
$message = sprintf('%s (column %d)', $message, $issue_data['column_from']);
|
|
|
|
$issue_string = sprintf(
|
|
|
|
'%s:%d: [%s] %s',
|
|
|
|
$issue_data['file_name'],
|
|
|
|
$issue_data['line_from'],
|
|
|
|
$code,
|
|
|
|
$message
|
|
|
|
);
|
2018-02-21 18:59:31 -05:00
|
|
|
|
2018-02-18 21:57:06 -08:00
|
|
|
return $issue_string;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array{severity: string, line_from: int, line_to: int, type: string, message: string,
|
|
|
|
* file_name: string, file_path: string, snippet: string, from: int, to: int,
|
|
|
|
* snippet_from: int, snippet_to: int, column_from: int, column_to: int} $issue_data
|
2018-06-04 10:19:20 -04:00
|
|
|
* @param bool $include_snippet
|
2017-07-25 16:11:02 -04:00
|
|
|
* @param bool $use_color
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2017-01-15 22:39:26 -05:00
|
|
|
* @return string
|
|
|
|
*/
|
2018-06-04 10:19:20 -04:00
|
|
|
protected static function getConsoleOutput(array $issue_data, $use_color, $include_snippet = true)
|
2017-01-15 22:39:26 -05:00
|
|
|
{
|
2017-07-25 16:11:02 -04:00
|
|
|
$issue_string = '';
|
2017-01-15 22:39:26 -05:00
|
|
|
|
2018-01-10 23:29:46 -05:00
|
|
|
$is_error = $issue_data['severity'] === Config::REPORT_ERROR;
|
|
|
|
|
|
|
|
if ($is_error) {
|
2017-07-25 16:11:02 -04:00
|
|
|
$issue_string .= ($use_color ? "\e[0;31mERROR\e[0m" : 'ERROR');
|
|
|
|
} else {
|
|
|
|
$issue_string .= 'INFO';
|
|
|
|
}
|
|
|
|
|
|
|
|
$issue_string .= ': ' . $issue_data['type'] . ' - ' . $issue_data['file_name'] . ':' .
|
2018-04-13 01:42:24 +02:00
|
|
|
$issue_data['line_from'] . ':' . $issue_data['column_from'] . ' - ' . $issue_data['message'] . "\n";
|
2017-07-25 16:11:02 -04:00
|
|
|
|
2018-06-04 10:19:20 -04:00
|
|
|
if ($include_snippet) {
|
|
|
|
$snippet = $issue_data['snippet'];
|
2017-07-25 16:11:02 -04:00
|
|
|
|
2018-06-04 10:19:20 -04:00
|
|
|
if (!$use_color) {
|
|
|
|
$issue_string .= $snippet;
|
|
|
|
} else {
|
|
|
|
$selection_start = $issue_data['from'] - $issue_data['snippet_from'];
|
|
|
|
$selection_length = $issue_data['to'] - $issue_data['from'];
|
2017-07-25 16:11:02 -04:00
|
|
|
|
2018-06-04 10:19:20 -04:00
|
|
|
$issue_string .= substr($snippet, 0, $selection_start)
|
|
|
|
. ($is_error ? "\e[97;41m" : "\e[30;47m") . substr($snippet, $selection_start, $selection_length)
|
|
|
|
. "\e[0m" . substr($snippet, $selection_length + $selection_start) . "\n";
|
|
|
|
}
|
2017-07-25 16:11:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return $issue_string;
|
2017-01-15 22:39:26 -05:00
|
|
|
}
|
|
|
|
|
2016-12-07 22:38:57 -05:00
|
|
|
/**
|
2018-09-25 18:37:24 -04:00
|
|
|
* @return array<int, array{severity: string, line_from: int, line_to: int, type: string, message: string,
|
|
|
|
* file_name: string, file_path: string, snippet: string, from: int, to: int, snippet_from: int, snippet_to: int,
|
2018-11-09 00:46:13 -05:00
|
|
|
* column_from: int, column_to: int, selected_text: string}>
|
2016-12-07 22:38:57 -05:00
|
|
|
*/
|
2017-07-25 16:11:02 -04:00
|
|
|
public static function getIssuesData()
|
2016-12-07 22:38:57 -05:00
|
|
|
{
|
2017-07-25 16:11:02 -04:00
|
|
|
return self::$issues_data;
|
2016-12-07 22:38:57 -05:00
|
|
|
}
|
|
|
|
|
2018-09-25 18:37:24 -04:00
|
|
|
/**
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
public static function getErrorCount()
|
|
|
|
{
|
|
|
|
return self::$error_count;
|
|
|
|
}
|
|
|
|
|
2016-12-07 22:38:57 -05:00
|
|
|
/**
|
2018-03-17 18:05:50 -04:00
|
|
|
* @param array<int, array{severity: string, line_from: int, line_to: int, type: string, message: string,
|
2017-07-25 16:11:02 -04:00
|
|
|
* file_name: string, file_path: string, snippet: string, from: int, to: int, snippet_from: int,
|
2018-02-19 18:16:09 -05:00
|
|
|
* snippet_to: int, column_from: int, column_to: int}> $issues_data
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2017-07-25 16:11:02 -04:00
|
|
|
* @return void
|
2016-12-07 22:38:57 -05:00
|
|
|
*/
|
2017-07-25 16:11:02 -04:00
|
|
|
public static function addIssues(array $issues_data)
|
2016-12-07 22:38:57 -05:00
|
|
|
{
|
2018-11-01 13:22:38 -04:00
|
|
|
foreach ($issues_data as $issue) {
|
2018-11-01 17:03:08 -04:00
|
|
|
$error_message = $issue['type']
|
|
|
|
. ' - ' . $issue['file_name']
|
|
|
|
. ':' . $issue['line_from']
|
|
|
|
. ' - ' . $issue['message'];
|
2018-11-01 13:22:38 -04:00
|
|
|
|
|
|
|
if (!self::alreadyEmitted($error_message)) {
|
|
|
|
self::$issues_data[] = $issue;
|
|
|
|
}
|
|
|
|
}
|
2016-12-07 22:38:57 -05:00
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2018-11-11 12:01:14 -05:00
|
|
|
* @param ProjectAnalyzer $project_analyzer
|
2018-10-30 15:32:20 +01:00
|
|
|
* @param bool $is_full
|
|
|
|
* @param float $start_time
|
|
|
|
* @param bool $add_stats
|
2018-11-09 00:46:13 -05:00
|
|
|
* @param array<string,array<string,array{o:int, s:array<int, string>}>> $issue_baseline
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2016-11-02 02:29:00 -04:00
|
|
|
* @return void
|
|
|
|
*/
|
2018-01-31 17:09:09 -05:00
|
|
|
public static function finish(
|
2018-11-11 12:01:14 -05:00
|
|
|
ProjectAnalyzer $project_analyzer,
|
2018-10-30 15:32:20 +01:00
|
|
|
bool $is_full,
|
|
|
|
float $start_time,
|
|
|
|
bool $add_stats = false,
|
|
|
|
array $issue_baseline = []
|
2018-01-31 17:09:09 -05:00
|
|
|
) {
|
2018-11-11 12:01:14 -05:00
|
|
|
if ($project_analyzer->output_format === ProjectAnalyzer::TYPE_CONSOLE) {
|
2018-04-16 14:05:37 -04:00
|
|
|
echo "\n";
|
|
|
|
}
|
2018-03-18 18:27:10 -04:00
|
|
|
|
2018-11-11 12:19:53 -05:00
|
|
|
$codebase = $project_analyzer->getCodebase();
|
|
|
|
|
2018-03-18 18:04:50 -04:00
|
|
|
$error_count = 0;
|
2018-03-18 18:27:10 -04:00
|
|
|
$info_count = 0;
|
2017-07-25 16:11:02 -04:00
|
|
|
|
|
|
|
if (self::$issues_data) {
|
2017-12-10 11:22:36 -05:00
|
|
|
usort(
|
|
|
|
self::$issues_data,
|
|
|
|
/** @return int */
|
|
|
|
function (array $d1, array $d2) {
|
|
|
|
if ($d1['file_path'] === $d2['file_path']) {
|
2018-02-19 18:16:09 -05:00
|
|
|
if ($d1['line_from'] === $d2['line_from']) {
|
|
|
|
if ($d1['column_from'] === $d2['column_from']) {
|
2017-12-10 11:22:36 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-02-19 18:16:09 -05:00
|
|
|
return $d1['column_from'] > $d2['column_from'] ? 1 : -1;
|
2017-12-10 11:22:36 -05:00
|
|
|
}
|
|
|
|
|
2018-02-19 18:16:09 -05:00
|
|
|
return $d1['line_from'] > $d2['line_from'] ? 1 : -1;
|
2017-12-10 11:22:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return $d1['file_path'] > $d2['file_path'] ? 1 : -1;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2018-10-30 15:32:20 +01:00
|
|
|
if (!empty($issue_baseline)) {
|
|
|
|
// Set severity for issues in baseline to INFO
|
|
|
|
foreach (self::$issues_data as $key => $issue_data) {
|
|
|
|
$file = $issue_data['file_name'];
|
|
|
|
$type = $issue_data['type'];
|
|
|
|
|
2018-11-09 00:46:13 -05:00
|
|
|
if (isset($issue_baseline[$file][$type]) && $issue_baseline[$file][$type]['o'] > 0) {
|
|
|
|
if ($issue_baseline[$file][$type]['o'] === count($issue_baseline[$file][$type]['s'])) {
|
|
|
|
$position = array_search($issue_data['selected_text'], $issue_baseline[$file][$type]['s']);
|
|
|
|
|
|
|
|
if ($position !== false) {
|
|
|
|
$issue_data['severity'] = Config::REPORT_INFO;
|
|
|
|
array_splice($issue_baseline[$file][$type]['s'], $position, 1);
|
|
|
|
$issue_baseline[$file][$type]['o'] = $issue_baseline[$file][$type]['o'] - 1;
|
|
|
|
}
|
|
|
|
} else {
|
2018-11-09 11:38:40 -05:00
|
|
|
$issue_baseline[$file][$type]['s'] = [];
|
2018-11-09 00:46:13 -05:00
|
|
|
$issue_data['severity'] = Config::REPORT_INFO;
|
|
|
|
$issue_baseline[$file][$type]['o'] = $issue_baseline[$file][$type]['o'] - 1;
|
|
|
|
}
|
2018-10-30 15:32:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
self::$issues_data[$key] = $issue_data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-08 17:18:48 +02:00
|
|
|
foreach (self::$issues_data as $issue_data) {
|
|
|
|
if ($issue_data['severity'] === Config::REPORT_ERROR) {
|
2018-03-18 18:04:50 -04:00
|
|
|
++$error_count;
|
2018-03-18 18:27:10 -04:00
|
|
|
} else {
|
|
|
|
++$info_count;
|
2017-07-25 16:11:02 -04:00
|
|
|
}
|
2017-09-08 17:18:48 +02:00
|
|
|
}
|
2017-07-25 16:11:02 -04:00
|
|
|
|
2018-06-04 10:19:20 -04:00
|
|
|
echo self::getOutput(
|
2018-11-11 12:01:14 -05:00
|
|
|
$project_analyzer->output_format,
|
|
|
|
$project_analyzer->use_color,
|
|
|
|
$project_analyzer->show_snippet,
|
|
|
|
$project_analyzer->show_info
|
2018-06-04 10:19:20 -04:00
|
|
|
);
|
2017-09-27 00:18:24 +02:00
|
|
|
}
|
2018-03-18 18:27:10 -04:00
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
foreach ($project_analyzer->reports as $format => $path) {
|
2017-09-27 00:18:24 +02:00
|
|
|
file_put_contents(
|
|
|
|
$path,
|
2018-11-11 12:01:14 -05:00
|
|
|
self::getOutput($format, $project_analyzer->use_color)
|
2017-09-27 00:18:24 +02:00
|
|
|
);
|
2017-07-25 16:11:02 -04:00
|
|
|
}
|
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
if ($project_analyzer->output_format === ProjectAnalyzer::TYPE_CONSOLE) {
|
2018-04-16 14:05:37 -04:00
|
|
|
echo str_repeat('-', 30) . "\n";
|
2018-03-18 18:04:50 -04:00
|
|
|
|
2018-04-16 14:05:37 -04:00
|
|
|
if ($error_count) {
|
2018-11-11 12:01:14 -05:00
|
|
|
echo ($project_analyzer->use_color
|
2018-04-16 14:05:37 -04:00
|
|
|
? "\e[0;31m" . $error_count . " errors\e[0m"
|
|
|
|
: $error_count . ' errors'
|
|
|
|
) . ' found' . "\n";
|
|
|
|
} else {
|
|
|
|
echo 'No errors found!' . "\n";
|
|
|
|
}
|
2018-03-18 18:27:10 -04:00
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
if ($info_count && $project_analyzer->show_info) {
|
2018-04-16 14:05:37 -04:00
|
|
|
echo str_repeat('-', 30) . "\n";
|
2018-03-18 18:27:10 -04:00
|
|
|
|
2018-04-16 14:05:37 -04:00
|
|
|
echo $info_count . ' other issues found.' . "\n"
|
|
|
|
. 'You can hide them with ' .
|
2018-11-11 12:01:14 -05:00
|
|
|
($project_analyzer->use_color
|
2018-04-16 14:05:37 -04:00
|
|
|
? "\e[30;48;5;195m--show-info=false\e[0m"
|
|
|
|
: '--show-info=false') . "\n";
|
|
|
|
}
|
2018-03-18 18:27:10 -04:00
|
|
|
|
2018-04-16 14:05:37 -04:00
|
|
|
echo str_repeat('-', 30) . "\n" . "\n";
|
2018-03-18 18:04:50 -04:00
|
|
|
|
2018-04-16 14:05:37 -04:00
|
|
|
if ($start_time) {
|
2018-10-10 16:05:06 -04:00
|
|
|
echo 'Checks took ' . number_format(microtime(true) - $start_time, 2) . ' seconds';
|
2018-04-16 14:05:37 -04:00
|
|
|
echo ' and used ' . number_format(memory_get_peak_usage() / (1024 * 1024), 3) . 'MB of memory' . "\n";
|
2018-01-31 16:08:52 -05:00
|
|
|
|
2018-04-16 14:05:37 -04:00
|
|
|
if ($is_full) {
|
2018-11-05 21:57:36 -05:00
|
|
|
$analysis_summary = $codebase->analyzer->getTypeInferenceSummary();
|
2018-06-03 18:31:43 -04:00
|
|
|
echo $analysis_summary . "\n";
|
2018-04-16 14:05:37 -04:00
|
|
|
}
|
2018-01-31 17:09:09 -05:00
|
|
|
|
2018-04-16 14:05:37 -04:00
|
|
|
if ($add_stats) {
|
|
|
|
echo '-----------------' . "\n";
|
2018-11-05 21:57:36 -05:00
|
|
|
echo $codebase->analyzer->getNonMixedStats();
|
2018-04-16 14:05:37 -04:00
|
|
|
echo "\n";
|
|
|
|
}
|
2018-01-31 16:08:52 -05:00
|
|
|
}
|
2016-12-14 19:24:16 -05:00
|
|
|
}
|
|
|
|
|
2018-03-18 18:04:50 -04:00
|
|
|
if ($error_count) {
|
2016-06-20 19:30:38 -04:00
|
|
|
exit(1);
|
|
|
|
}
|
2016-10-07 00:58:08 -04:00
|
|
|
|
2016-11-06 00:59:29 -04:00
|
|
|
if ($is_full && $start_time) {
|
2018-11-11 12:19:53 -05:00
|
|
|
$codebase->file_reference_provider->removeDeletedFilesFromReferences();
|
2018-09-28 16:18:45 -04:00
|
|
|
|
2018-11-11 12:19:53 -05:00
|
|
|
if ($codebase->statements_provider->parser_cache_provider) {
|
|
|
|
$codebase->statements_provider->parser_cache_provider->processSuccessfulRun($start_time);
|
2018-09-28 16:18:45 -04:00
|
|
|
}
|
2016-10-07 00:58:08 -04:00
|
|
|
}
|
2016-06-20 19:30:38 -04:00
|
|
|
}
|
2016-10-18 18:55:53 -04:00
|
|
|
|
2016-10-30 12:46:18 -04:00
|
|
|
/**
|
2017-09-08 17:18:48 +02:00
|
|
|
* @param string $format
|
2018-06-04 10:19:20 -04:00
|
|
|
* @param bool $use_color
|
|
|
|
* @param bool $show_snippet
|
2018-09-26 18:08:23 -04:00
|
|
|
* @param bool $show_info
|
2017-09-08 17:18:48 +02:00
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2018-09-26 18:08:23 -04:00
|
|
|
public static function getOutput($format, $use_color, $show_snippet = true, $show_info = true)
|
2017-09-08 17:18:48 +02:00
|
|
|
{
|
2018-11-05 21:57:36 -05:00
|
|
|
if ($format === ProjectAnalyzer::TYPE_JSON) {
|
2018-04-13 01:42:24 +02:00
|
|
|
return json_encode(self::$issues_data) . "\n";
|
2018-11-05 21:57:36 -05:00
|
|
|
} elseif ($format === ProjectAnalyzer::TYPE_XML) {
|
2017-09-08 17:18:48 +02:00
|
|
|
$xml = Array2XML::createXML('report', ['item' => self::$issues_data]);
|
|
|
|
|
|
|
|
return $xml->saveXML();
|
2018-11-05 21:57:36 -05:00
|
|
|
} elseif ($format === ProjectAnalyzer::TYPE_EMACS) {
|
2017-09-08 17:18:48 +02:00
|
|
|
$output = '';
|
|
|
|
foreach (self::$issues_data as $issue_data) {
|
2018-04-13 01:42:24 +02:00
|
|
|
$output .= self::getEmacsOutput($issue_data) . "\n";
|
2017-09-08 17:18:48 +02:00
|
|
|
}
|
|
|
|
|
2018-02-18 21:57:06 -08:00
|
|
|
return $output;
|
2018-11-05 21:57:36 -05:00
|
|
|
} elseif ($format === ProjectAnalyzer::TYPE_PYLINT) {
|
2018-02-18 21:57:06 -08:00
|
|
|
$output = '';
|
|
|
|
foreach (self::$issues_data as $issue_data) {
|
2018-04-13 01:42:24 +02:00
|
|
|
$output .= self::getPylintOutput($issue_data) . "\n";
|
2018-02-18 21:57:06 -08:00
|
|
|
}
|
|
|
|
|
2017-09-08 17:18:48 +02:00
|
|
|
return $output;
|
|
|
|
}
|
|
|
|
|
|
|
|
$output = '';
|
|
|
|
foreach (self::$issues_data as $issue_data) {
|
2018-09-26 18:08:23 -04:00
|
|
|
if (!$show_info && $issue_data['severity'] === Config::REPORT_INFO) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-06-04 10:19:20 -04:00
|
|
|
$output .= self::getConsoleOutput($issue_data, $use_color, $show_snippet) . "\n" . "\n";
|
2017-09-08 17:18:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $output;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-10-30 12:46:18 -04:00
|
|
|
* @param string $message
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2016-10-30 12:46:18 -04:00
|
|
|
* @return bool
|
|
|
|
*/
|
2016-10-18 18:55:53 -04:00
|
|
|
protected static function alreadyEmitted($message)
|
|
|
|
{
|
|
|
|
$sham = sha1($message);
|
|
|
|
|
|
|
|
if (isset(self::$emitted[$sham])) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
self::$emitted[$sham] = true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2016-12-08 15:57:18 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function clearCache()
|
|
|
|
{
|
2017-07-25 16:11:02 -04:00
|
|
|
self::$issues_data = [];
|
2016-12-08 15:57:18 -05:00
|
|
|
self::$emitted = [];
|
2017-01-14 01:24:27 -05:00
|
|
|
self::$error_count = 0;
|
2017-03-13 18:06:56 -04:00
|
|
|
self::$recording_level = 0;
|
|
|
|
self::$recorded_issues = [];
|
2017-07-25 16:11:02 -04:00
|
|
|
self::$console_issues = [];
|
2017-03-13 18:06:56 -04:00
|
|
|
}
|
|
|
|
|
2018-10-17 15:52:26 -04:00
|
|
|
/**
|
|
|
|
* @return array<int, array{severity: string, line_from: int, line_to: int, type: string, message: string,
|
|
|
|
* file_name: string, file_path: string, snippet: string, from: int, to: int, snippet_from: int, snippet_to: int,
|
|
|
|
* column_from: int, column_to: int}>
|
|
|
|
*/
|
|
|
|
public static function clear()
|
|
|
|
{
|
|
|
|
$current_data = self::$issues_data;
|
|
|
|
self::$issues_data = [];
|
|
|
|
self::$emitted = [];
|
|
|
|
return $current_data;
|
|
|
|
}
|
|
|
|
|
2017-06-29 00:28:37 -04:00
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public static function isRecording()
|
|
|
|
{
|
|
|
|
return self::$recording_level > 0;
|
|
|
|
}
|
|
|
|
|
2017-03-13 18:06:56 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function startRecording()
|
|
|
|
{
|
2017-05-26 20:05:57 -04:00
|
|
|
++self::$recording_level;
|
2017-03-13 18:06:56 -04:00
|
|
|
self::$recorded_issues[self::$recording_level] = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function stopRecording()
|
|
|
|
{
|
|
|
|
if (self::$recording_level === 0) {
|
|
|
|
throw new \UnexpectedValueException('Cannot stop recording - already at base level');
|
|
|
|
}
|
|
|
|
|
2017-05-26 20:05:57 -04:00
|
|
|
--self::$recording_level;
|
2017-03-13 18:06:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array<int, CodeIssue>
|
|
|
|
*/
|
|
|
|
public static function clearRecordingLevel()
|
|
|
|
{
|
|
|
|
if (self::$recording_level === 0) {
|
|
|
|
throw new \UnexpectedValueException('Not currently recording');
|
|
|
|
}
|
|
|
|
|
|
|
|
$recorded_issues = self::$recorded_issues[self::$recording_level];
|
|
|
|
|
|
|
|
self::$recorded_issues[self::$recording_level] = [];
|
|
|
|
|
|
|
|
return $recorded_issues;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function bubbleUp(CodeIssue $e)
|
|
|
|
{
|
|
|
|
if (self::$recording_level === 0) {
|
|
|
|
self::add($e);
|
2017-05-24 22:07:49 -04:00
|
|
|
|
2017-03-13 18:06:56 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
self::$recorded_issues[self::$recording_level][] = $e;
|
2016-12-08 15:57:18 -05:00
|
|
|
}
|
2016-06-05 20:25:16 -04:00
|
|
|
}
|