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
|
|
|
|
2016-08-13 14:20:46 -04:00
|
|
|
use Psalm\Checker\ProjectChecker;
|
2017-03-13 18:06:56 -04:00
|
|
|
use Psalm\Issue\CodeIssue;
|
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
|
|
|
{
|
2016-12-07 22:38:57 -05:00
|
|
|
/**
|
|
|
|
* @var array<int, array>
|
|
|
|
*/
|
|
|
|
protected static $issue_data = [];
|
|
|
|
|
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-12-14 19:24:16 -05:00
|
|
|
/**
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
protected static $start_time = 0;
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2017-03-13 18:06:56 -04:00
|
|
|
* @param CodeIssue $e
|
|
|
|
* @param array $suppressed_issues
|
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
|
|
|
|
|
|
|
if (in_array($issue_type, $suppressed_issues)) {
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2017-03-13 18:06:56 -04:00
|
|
|
* @param CodeIssue $e
|
2016-11-02 02:29:00 -04:00
|
|
|
* @return bool
|
|
|
|
* @throws Exception\CodeException
|
|
|
|
*/
|
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-12-07 22:38:57 -05:00
|
|
|
$project_checker = ProjectChecker::getInstance();
|
2016-06-27 13:22:16 -04:00
|
|
|
|
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
|
|
|
|
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
|
|
|
|
2016-12-03 19:11:30 -05:00
|
|
|
if ($reporting_level === Config::REPORT_SUPPRESS) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($reporting_level === Config::REPORT_INFO) {
|
2016-12-07 22:38:57 -05:00
|
|
|
if ($project_checker->show_info && !self::alreadyEmitted($error_message)) {
|
|
|
|
switch ($project_checker->output_format) {
|
|
|
|
case ProjectChecker::TYPE_CONSOLE:
|
|
|
|
echo 'INFO: ' . $error_message . PHP_EOL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ProjectChecker::TYPE_JSON:
|
|
|
|
self::$issue_data[] = self::getIssueArray($e, Config::REPORT_INFO);
|
|
|
|
break;
|
2017-01-15 22:39:26 -05:00
|
|
|
|
|
|
|
case ProjectChecker::TYPE_EMACS:
|
|
|
|
echo self::getEmacsOutput($e, Config::REPORT_INFO) . PHP_EOL;
|
|
|
|
break;
|
2016-12-07 22:38:57 -05:00
|
|
|
}
|
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)) {
|
2016-12-07 22:38:57 -05:00
|
|
|
switch ($project_checker->output_format) {
|
|
|
|
case ProjectChecker::TYPE_CONSOLE:
|
2017-05-26 23:26:14 +01:00
|
|
|
$error_maybe_with_color = ($project_checker->use_color ? "\e[0;31mERROR\e[0m" : 'ERROR');
|
|
|
|
echo $error_maybe_with_color . ': ' . $error_message . PHP_EOL;
|
2016-12-07 22:38:57 -05:00
|
|
|
|
|
|
|
echo self::getSnippet($e, $project_checker->use_color) . PHP_EOL . PHP_EOL;
|
2017-01-14 01:24:27 -05:00
|
|
|
self::$error_count++;
|
2016-12-03 19:11:30 -05:00
|
|
|
|
2016-12-07 22:38:57 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ProjectChecker::TYPE_JSON:
|
|
|
|
self::$issue_data[] = self::getIssueArray($e);
|
|
|
|
break;
|
2017-01-15 22:39:26 -05:00
|
|
|
|
|
|
|
case ProjectChecker::TYPE_EMACS:
|
|
|
|
echo self::getEmacsOutput($e) . PHP_EOL;
|
|
|
|
break;
|
2016-12-07 22:38:57 -05:00
|
|
|
}
|
2016-10-18 18:55:53 -04:00
|
|
|
}
|
2016-06-16 19:02:29 -04:00
|
|
|
|
2016-06-20 19:30:38 -04:00
|
|
|
if ($config->stop_on_first_error) {
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
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
|
|
|
/**
|
2017-03-13 18:06:56 -04:00
|
|
|
* @param CodeIssue $e
|
2016-12-07 22:38:57 -05:00
|
|
|
* @param string $severity
|
|
|
|
* @return array
|
|
|
|
*/
|
2017-03-13 18:06:56 -04:00
|
|
|
protected static function getIssueArray(CodeIssue $e, $severity = Config::REPORT_ERROR)
|
2016-12-07 22:38:57 -05:00
|
|
|
{
|
|
|
|
$location = $e->getLocation();
|
|
|
|
$selection_bounds = $location->getSelectionBounds();
|
|
|
|
|
|
|
|
return [
|
|
|
|
'type' => $severity,
|
|
|
|
'line_number' => $location->getLineNumber(),
|
|
|
|
'message' => $e->getMessage(),
|
|
|
|
'file_name' => $location->file_name,
|
|
|
|
'file_path' => $location->file_path,
|
|
|
|
'snippet' => $location->getSnippet(),
|
|
|
|
'from' => $selection_bounds[0],
|
|
|
|
'to' => $selection_bounds[1],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2017-01-15 22:39:26 -05:00
|
|
|
/**
|
2017-03-13 18:06:56 -04:00
|
|
|
* @param CodeIssue $e
|
2017-01-15 22:39:26 -05:00
|
|
|
* @param string $severity
|
|
|
|
* @return string
|
|
|
|
*/
|
2017-03-13 18:06:56 -04:00
|
|
|
protected static function getEmacsOutput(CodeIssue $e, $severity = Config::REPORT_ERROR)
|
2017-01-15 22:39:26 -05:00
|
|
|
{
|
|
|
|
$location = $e->getLocation();
|
|
|
|
|
|
|
|
return $location->file_path . ':' . $location->getLineNumber() . ':' . $location->getColumn() . ': ' .
|
|
|
|
($severity === Config::REPORT_ERROR ? 'error' : 'warning') . ' - ' . $e->getMessage();
|
|
|
|
}
|
|
|
|
|
2016-12-07 22:38:57 -05:00
|
|
|
/**
|
|
|
|
* @return array<int, array>
|
|
|
|
*/
|
|
|
|
public static function getIssueData()
|
|
|
|
{
|
|
|
|
return self::$issue_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-03-13 18:06:56 -04:00
|
|
|
* @param CodeIssue $e
|
2016-12-07 22:38:57 -05:00
|
|
|
* @param boolean $use_color
|
|
|
|
* @return string
|
|
|
|
*/
|
2017-03-13 18:06:56 -04:00
|
|
|
protected static function getSnippet(CodeIssue $e, $use_color = true)
|
2016-12-07 22:38:57 -05:00
|
|
|
{
|
|
|
|
$location = $e->getLocation();
|
|
|
|
|
|
|
|
$snippet = $location->getSnippet();
|
|
|
|
|
|
|
|
if (!$use_color) {
|
|
|
|
return $snippet;
|
|
|
|
}
|
|
|
|
|
|
|
|
$snippet_bounds = $location->getSnippetBounds();
|
|
|
|
$selection_bounds = $location->getSelectionBounds();
|
|
|
|
|
|
|
|
$selection_start = $selection_bounds[0] - $snippet_bounds[0];
|
2016-12-08 15:57:18 -05:00
|
|
|
$selection_length = $selection_bounds[1] - $selection_bounds[0];
|
2016-12-07 22:38:57 -05:00
|
|
|
|
|
|
|
return substr($snippet, 0, $selection_start) .
|
|
|
|
"\e[97;41m" . substr($snippet, $selection_start, $selection_length) .
|
|
|
|
"\e[0m" . substr($snippet, $selection_length + $selection_start) . PHP_EOL;
|
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2017-02-18 13:41:27 -05:00
|
|
|
* @param bool $is_full
|
|
|
|
* @param int|null $start_time
|
|
|
|
* @param array<string, bool> $visited_files
|
2016-11-02 02:29:00 -04:00
|
|
|
* @return void
|
|
|
|
*/
|
2017-02-18 13:41:27 -05:00
|
|
|
public static function finish($is_full, $start_time, array $visited_files)
|
2016-06-20 19:30:38 -04:00
|
|
|
{
|
2017-02-18 13:41:27 -05:00
|
|
|
Provider\FileReferenceProvider::updateReferenceCache($visited_files);
|
2016-10-05 13:24:46 -04:00
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
if ($start_time) {
|
2016-12-24 11:03:55 +00:00
|
|
|
echo('Checks took ' . ((float)microtime(true) - self::$start_time));
|
2017-01-06 01:07:11 -05:00
|
|
|
echo(' and used ' . number_format(memory_get_peak_usage() / (1024 * 1024), 3) . 'MB' . PHP_EOL);
|
2016-12-14 19:24:16 -05:00
|
|
|
}
|
|
|
|
|
2017-01-14 01:24:27 -05:00
|
|
|
if (self::$error_count) {
|
2016-12-07 22:38:57 -05:00
|
|
|
$project_checker = ProjectChecker::getInstance();
|
|
|
|
if ($project_checker->output_format === ProjectChecker::TYPE_JSON) {
|
|
|
|
echo json_encode(self::$issue_data) . PHP_EOL;
|
|
|
|
}
|
|
|
|
|
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) {
|
2017-02-18 13:41:27 -05:00
|
|
|
Provider\CacheProvider::processSuccessfulRun($start_time);
|
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
|
|
|
/**
|
|
|
|
* @param string $message
|
|
|
|
* @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
|
|
|
|
2016-12-14 19:24:16 -05:00
|
|
|
/**
|
2016-12-17 00:48:31 -05:00
|
|
|
* @param int $time
|
2016-12-14 19:24:16 -05:00
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function setStartTime($time)
|
|
|
|
{
|
|
|
|
self::$start_time = $time;
|
|
|
|
}
|
|
|
|
|
2016-12-08 15:57:18 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function clearCache()
|
|
|
|
{
|
|
|
|
self::$issue_data = [];
|
|
|
|
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 = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function startRecording()
|
|
|
|
{
|
|
|
|
self::$recording_level++;
|
|
|
|
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');
|
|
|
|
}
|
|
|
|
|
|
|
|
self::$recording_level--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @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
|
|
|
}
|