1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-22 13:51:54 +01:00
psalm/src/psalm.php

717 lines
21 KiB
PHP
Raw Normal View History

2016-07-26 15:12:44 -04:00
<?php
require_once('command_functions.php');
2016-07-26 15:12:44 -04:00
use Psalm\ErrorBaseline;
2018-11-05 21:57:36 -05:00
use Psalm\Internal\Analyzer\ProjectAnalyzer;
use Psalm\Internal\Provider;
2017-10-07 14:41:16 -04:00
use Psalm\Config;
use Psalm\IssueBuffer;
2017-10-07 14:41:16 -04:00
2016-07-26 15:12:44 -04:00
// show all errors
2016-10-30 12:46:18 -04:00
error_reporting(-1);
2016-07-26 15:12:44 -04:00
$valid_short_options = [
'f:',
'm',
'h',
'v',
'c:',
'i',
'r:',
];
$valid_long_options = [
'clear-cache',
2018-10-15 11:29:57 -04:00
'clear-global-cache',
'config:',
'debug',
'debug-by-line',
'diff',
'diff-methods',
'disable-extension:',
'find-dead-code::',
'find-references-to:',
'help',
'ignore-baseline',
'init',
'monochrome',
'no-cache',
'no-reflection-cache',
'output-format:',
'plugin:',
'report:',
'root:',
'set-baseline:',
'show-info:',
'show-snippet:',
'stats',
'threads:',
'update-baseline',
'use-ini-defaults',
'version',
'php-version:',
2019-02-24 01:33:25 -05:00
'generate-json-map:',
];
gc_collect_cycles();
gc_disable();
$args = array_slice($argv, 1);
array_map(
/**
* @param string $arg
*
* @return void
*/
function ($arg) use ($valid_long_options, $valid_short_options) {
if (substr($arg, 0, 2) === '--' && $arg !== '--') {
$arg_name = preg_replace('/=.*$/', '', substr($arg, 2));
if (!in_array($arg_name, $valid_long_options)
&& !in_array($arg_name . ':', $valid_long_options)
&& !in_array($arg_name . '::', $valid_long_options)
) {
echo 'Unrecognised argument "--' . $arg_name . '"' . PHP_EOL
. 'Type --help to see a list of supported arguments'. PHP_EOL;
exit(1);
}
} elseif (substr($arg, 0, 2) === '-' && $arg !== '-' && $arg !== '--') {
$arg_name = preg_replace('/=.*$/', '', substr($arg, 1));
if (!in_array($arg_name, $valid_short_options) && !in_array($arg_name . ':', $valid_short_options)) {
echo 'Unrecognised argument "-' . $arg_name . '"' . PHP_EOL
. 'Type --help to see a list of supported arguments'. PHP_EOL;
exit(1);
}
}
},
$args
2016-12-07 22:38:57 -05:00
);
// get options from command line
$options = getopt(implode('', $valid_short_options), $valid_long_options);
if (!array_key_exists('use-ini-defaults', $options)) {
ini_set('display_errors', '1');
ini_set('display_startup_errors', '1');
ini_set('memory_limit', (string) (4 * 1024 * 1024 * 1024));
}
if (array_key_exists('help', $options)) {
2016-11-04 22:55:13 -04:00
$options['h'] = false;
}
if (array_key_exists('version', $options)) {
$options['v'] = false;
}
2017-02-12 23:59:33 -05:00
if (array_key_exists('init', $options)) {
$options['i'] = false;
}
if (array_key_exists('monochrome', $options)) {
2016-11-04 22:55:13 -04:00
$options['m'] = false;
}
if (isset($options['config'])) {
2016-11-04 22:55:13 -04:00
$options['c'] = $options['config'];
}
2017-01-16 19:44:31 -05:00
if (isset($options['c']) && is_array($options['c'])) {
echo 'Too many config files provided' . PHP_EOL;
exit(1);
2017-01-16 19:44:31 -05:00
}
if (array_key_exists('h', $options)) {
echo <<<HELP
Usage:
2016-11-04 22:55:13 -04:00
psalm [options] [file...]
Options:
2017-02-12 23:59:33 -05:00
-h, --help
Display this help message
-v, --version
Display the Psalm version
-i, --init [source_dir=src] [level=3]
2017-02-12 23:59:33 -05:00
Create a psalm config file in the current directory that points to [source_dir]
at the required level, from 1, most strict, to 8, most permissive.
2017-02-12 23:59:33 -05:00
--debug
Debug information
2018-03-26 09:08:55 -04:00
--debug-by-line
Debug information on a line-by-line level
2017-02-12 23:59:33 -05:00
-c, --config=psalm.xml
Path to a psalm.xml configuration file. Run psalm --init to create one.
-m, --monochrome
Enable monochrome output
-r, --root
If running Psalm globally you'll need to specify a project root. Defaults to cwd
2017-02-12 23:59:33 -05:00
--show-info[=BOOLEAN]
Show non-exception parser findings
2018-06-04 10:19:20 -04:00
--show-snippet[=true]
Show code snippets with errors. Options are 'true' or 'false'
2017-02-12 23:59:33 -05:00
--diff
Runs Psalm in diff mode, only checking files that have changed (and their dependents)
--diff-methods
Only checks methods that have changed (and their dependents)
2017-02-12 23:59:33 -05:00
--output-format=console
Changes the output format. Possible values: compact, console, emacs, json, pylint, xml
2017-02-12 23:59:33 -05:00
--find-dead-code[=auto]
Look for dead code. Options are 'auto' or 'always'. If no value is specified, default is 'auto'
2019-01-22 11:10:37 -05:00
--find-references-to=[class|method|property]
Searches the codebase for references to the given fully-qualified class or method,
where method is in the format class::methodName
Refactor scanning and analysis, introducing multithreading (#191) * Add failing test * Add visitor to soup up classlike references * Move a whole bunch of code into the visitor * Move some methods back, move onto analysis stage * Use the getAliases method everywhere * Fix refs * Fix more refs * Fix some tests * Fix more tests * Fix include tests * Shift config class finding to project checker and fix bugs * Fix a few more tests * transition test to new syntax * Remove var_dump * Delete a bunch of code and fix mutation test * Remove unnecessary visitation * Transition to better mocked out file provider, breaking some cached statement loading * Use different scheme for naming anonymous classes * Fix anonymous class issues * Refactor file/statement loading * Add specific property types * Fix mapped property assignment * Improve how we deal with traits * Fix trait checking * Pass Psalm checks * Add multi-process support * Delay console output until the end * Remove PHP 7 syntax * Update file storage with classes * Fix scanning individual files and add reflection return types * Always turn XDebug off * Add quicker method of getting method mutations * Queue return types for crawling * Interpret all strings as possible classes once we see a `get_class` call * Check invalid return types again * Fix template namespacing issues * Default to class-insensitive file names for includes * Don’t overwrite existing issues data * Add var docblocks for scanning * Add null check * Fix loading of external classes in templates * Only try to populate class when we haven’t yet seen it’s not a class * Fix trait property accessibility * Only ever improve docblock param type * Make param replacement more robust * Fix static const missing inferred type * Fix a few more tests * Register constant definitions * Fix trait aliasing * Skip constant type tests for now * Fix linting issues * Make sure caching is off for tests * Remove unnecessary return * Use emulative parser if on PHP 5.6 * Cache parser for faster first-time parse * Fix constant resolution when scanning classes * Remove test that’s beyond a practical scope * Add back --diff support * Add --help for --threads * Remove unused vars
2017-07-25 16:11:02 -04:00
--threads=INT
If greater than one, Psalm will run analysis on multiple threads, speeding things up.
--report=PATH
The path where to output report file. The output format is base on the file extension.
(Currently supported format: ".json", ".xml", ".txt", ".emacs")
2017-10-07 14:41:16 -04:00
--clear-cache
2018-10-15 11:29:57 -04:00
Clears all cache files that Psalm uses for this specific project
--clear-global-cache
Clears all cache files that Psalm uses for all projects
2017-10-07 14:41:16 -04:00
2017-10-13 21:27:20 -04:00
--no-cache
Runs Psalm without using cache
--no-reflection-cache
Runs Psalm without using cached representations of unchanged classes and files.
Useful if you want the afterClassLikeVisit plugin hook to run every time you visit a file.
--plugin=PATH
Executes a plugin, an alternative to using the Psalm config
2018-01-31 17:09:09 -05:00
--stats
Shows a breakdown of Psalm's ability to infer types in the codebase
--use-ini-defaults
Use PHP-provided ini defaults for memory and error display
2018-07-17 16:54:34 -04:00
--disable-extension=[extension]
Used to disable certain extensions while Psalm is running.
--set-baseline=PATH
Save all current error level issues to a file, to mark them as info in subsequent runs
2018-11-01 11:15:08 -04:00
--ignore-baseline
Ignore the error baseline
--update-baseline
Update the baseline by removing fixed issues. This will not add new issues to the baseline
2018-07-17 16:54:34 -04:00
2019-02-24 01:33:25 -05:00
--generate-json-map=PATH
Generate a map of node references and types in JSON format, saved to the given path.
HELP;
exit;
}
2016-07-26 15:12:44 -04:00
if (getcwd() === false) {
echo 'Cannot get current working directory' . PHP_EOL;
exit(1);
}
if (isset($options['root'])) {
$options['r'] = $options['root'];
}
$current_dir = (string)getcwd() . DIRECTORY_SEPARATOR;
if (isset($options['r']) && is_string($options['r'])) {
$root_path = realpath($options['r']);
if (!$root_path) {
echo 'Could not locate root directory ' . $current_dir . DIRECTORY_SEPARATOR . $options['r'] . PHP_EOL;
exit(1);
}
$current_dir = $root_path . DIRECTORY_SEPARATOR;
}
$vendor_dir = getVendorDir($current_dir);
$first_autoloader = requireAutoloaders($current_dir, isset($options['r']), $vendor_dir);
if (array_key_exists('v', $options)) {
echo 'Psalm ' . PSALM_VERSION . PHP_EOL;
exit;
}
$threads = isset($options['threads']) ? (int)$options['threads'] : 1;
2018-11-05 21:57:36 -05:00
$ini_handler = new \Psalm\Internal\Fork\PsalmRestarter('PSALM');
2018-07-17 16:54:34 -04:00
if (isset($options['disable-extension'])) {
if (is_array($options['disable-extension'])) {
/** @psalm-suppress MixedAssignment */
2018-07-17 16:54:34 -04:00
foreach ($options['disable-extension'] as $extension) {
if (is_string($extension)) {
$ini_handler->disableExtension($extension);
}
}
} elseif (is_string($options['disable-extension'])) {
$ini_handler->disableExtension($options['disable-extension']);
}
}
if ($threads > 1) {
$ini_handler->disableExtension('grpc');
}
2019-02-24 01:33:25 -05:00
$type_map_location = null;
if (isset($options['generate-type-map']) && is_string($options['generate-type-map'])) {
$type_map_location = $options['generate-type-map'];
}
2017-10-07 14:41:16 -04:00
// If XDebug is enabled, restart without it
$ini_handler->check();
2017-10-07 14:41:16 -04:00
2018-02-17 22:11:42 -05:00
setlocale(LC_CTYPE, 'C');
if (isset($options['set-baseline'])) {
if (is_array($options['set-baseline'])) {
die('Only one baseline file can be created at a time' . PHP_EOL);
}
}
2017-02-12 23:59:33 -05:00
if (isset($options['i'])) {
if (file_exists($current_dir . 'psalm.xml')) {
2017-02-12 23:59:33 -05:00
die('A config file already exists in the current directory' . PHP_EOL);
}
$args = array_values(array_filter(
$args,
/**
* @param string $arg
*
* @return bool
*/
function ($arg) {
return $arg !== '--ansi'
&& $arg !== '--no-ansi'
&& $arg !== '-i'
&& $arg !== '--init'
&& strpos($arg, '--root=') !== 0
&& strpos($arg, '--r=') !== 0;
}
));
2017-02-12 23:59:33 -05:00
$level = 3;
$source_dir = 'src';
if (count($args)) {
if (count($args) > 2) {
die('Too many arguments provided for psalm --init' . PHP_EOL);
}
if (isset($args[1])) {
if (!preg_match('/^[1-8]$/', $args[1])) {
die('Config strictness must be a number between 1 and 8 inclusive' . PHP_EOL);
2017-02-12 23:59:33 -05:00
}
$level = (int)$args[1];
}
2017-02-13 00:12:56 -05:00
$source_dir = $args[0];
}
if (!is_dir($source_dir)) {
$bad_dir_path = getcwd() . DIRECTORY_SEPARATOR . $source_dir;
if (!isset($args[0])) {
die('Please specify a directory - the default, "src", was not found in this project.' . PHP_EOL);
2017-02-12 23:59:33 -05:00
}
2017-02-13 00:12:56 -05:00
die('The given path "' . $bad_dir_path . '" does not appear to be a directory' . PHP_EOL);
2017-02-12 23:59:33 -05:00
}
2017-05-28 00:07:26 -04:00
$template_file_name = dirname(__DIR__) . '/assets/config_levels/' . $level . '.xml';
2017-02-12 23:59:33 -05:00
if (!file_exists($template_file_name)) {
2017-05-28 00:07:26 -04:00
die('Could not open config template ' . $template_file_name . PHP_EOL);
2017-02-12 23:59:33 -05:00
}
$template = (string)file_get_contents($template_file_name);
$template = str_replace(
'<directory name="src" />',
'<directory name="' . $source_dir . '" />',
$template
);
2017-02-12 23:59:33 -05:00
if (!\Phar::running(false)) {
$template = str_replace(
'vendor/vimeo/psalm/config.xsd',
2018-04-22 00:50:58 +02:00
'file://' . realpath(__DIR__ . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'config.xsd'),
$template
);
}
if (!file_put_contents($current_dir . 'psalm.xml', $template)) {
2017-02-12 23:59:33 -05:00
die('Could not write to psalm.xml' . PHP_EOL);
}
exit('Config file created successfully. Please re-run psalm.' . PHP_EOL);
}
$output_format = isset($options['output-format']) && is_string($options['output-format'])
? $options['output-format']
2018-11-05 21:57:36 -05:00
: ProjectAnalyzer::TYPE_CONSOLE;
2016-12-07 22:38:57 -05:00
$paths_to_check = getPathsToCheck(isset($options['f']) ? $options['f'] : null);
$plugins = [];
if (isset($options['plugin'])) {
$plugins = $options['plugin'];
if (!is_array($plugins)) {
$plugins = [$plugins];
}
}
$path_to_config = isset($options['c']) && is_string($options['c']) ? realpath($options['c']) : null;
2016-11-04 22:55:13 -04:00
if ($path_to_config === false) {
/** @psalm-suppress InvalidCast */
echo 'Could not resolve path to config ' . (string)$options['c'] . PHP_EOL;
exit(1);
}
$show_info = isset($options['show-info'])
? $options['show-info'] !== 'false' && $options['show-info'] !== '0'
: true;
2016-11-04 22:55:13 -04:00
2016-10-07 00:58:08 -04:00
$is_diff = isset($options['diff']);
2016-07-26 15:12:44 -04:00
/** @var false|'always'|'auto' $find_dead_code */
$find_dead_code = false;
if (isset($options['find-dead-code'])) {
if ($options['find-dead-code'] === 'always') {
$find_dead_code = 'always';
} else {
$find_dead_code = 'auto';
}
}
$find_references_to = isset($options['find-references-to']) && is_string($options['find-references-to'])
? $options['find-references-to']
: null;
// initialise custom config, if passed
2018-04-19 17:29:07 -04:00
try {
if ($path_to_config) {
$config = Config::loadFromXMLFile($path_to_config, $current_dir);
} else {
$config = Config::getConfigForPath($current_dir, $current_dir, $output_format);
}
} catch (Psalm\Exception\ConfigException $e) {
echo $e->getMessage();
exit(1);
}
$config->setComposerClassLoader($first_autoloader);
2018-02-07 20:15:56 -05:00
if (isset($options['clear-cache'])) {
$cache_directory = $config->getCacheDirectory();
Config::removeCacheDirectory($cache_directory);
echo 'Cache directory deleted' . PHP_EOL;
exit;
}
2018-10-15 11:29:57 -04:00
if (isset($options['clear-global-cache'])) {
$cache_directory = $config->getGlobalCacheDirectory();
Config::removeCacheDirectory($cache_directory);
echo 'Global cache directory deleted' . PHP_EOL;
exit;
}
$debug = array_key_exists('debug', $options) || array_key_exists('debug-by-line', $options);
if (isset($options['no-cache'])) {
$providers = new Provider\Providers(
new Provider\FileProvider
);
} else {
$no_reflection_cache = isset($options['no-reflection-cache']);
$file_storage_cache_provider = $no_reflection_cache
? null
: new Provider\FileStorageCacheProvider($config);
$classlike_storage_cache_provider = $no_reflection_cache
? null
: new Provider\ClassLikeStorageCacheProvider($config);
$providers = new Provider\Providers(
new Provider\FileProvider,
new Provider\ParserCacheProvider($config),
$file_storage_cache_provider,
$classlike_storage_cache_provider,
new Provider\FileReferenceCacheProvider($config)
);
}
2018-11-11 12:01:14 -05:00
$project_analyzer = new ProjectAnalyzer(
$config,
$providers,
!array_key_exists('m', $options),
$show_info,
$output_format,
Refactor scanning and analysis, introducing multithreading (#191) * Add failing test * Add visitor to soup up classlike references * Move a whole bunch of code into the visitor * Move some methods back, move onto analysis stage * Use the getAliases method everywhere * Fix refs * Fix more refs * Fix some tests * Fix more tests * Fix include tests * Shift config class finding to project checker and fix bugs * Fix a few more tests * transition test to new syntax * Remove var_dump * Delete a bunch of code and fix mutation test * Remove unnecessary visitation * Transition to better mocked out file provider, breaking some cached statement loading * Use different scheme for naming anonymous classes * Fix anonymous class issues * Refactor file/statement loading * Add specific property types * Fix mapped property assignment * Improve how we deal with traits * Fix trait checking * Pass Psalm checks * Add multi-process support * Delay console output until the end * Remove PHP 7 syntax * Update file storage with classes * Fix scanning individual files and add reflection return types * Always turn XDebug off * Add quicker method of getting method mutations * Queue return types for crawling * Interpret all strings as possible classes once we see a `get_class` call * Check invalid return types again * Fix template namespacing issues * Default to class-insensitive file names for includes * Don’t overwrite existing issues data * Add var docblocks for scanning * Add null check * Fix loading of external classes in templates * Only try to populate class when we haven’t yet seen it’s not a class * Fix trait property accessibility * Only ever improve docblock param type * Make param replacement more robust * Fix static const missing inferred type * Fix a few more tests * Register constant definitions * Fix trait aliasing * Skip constant type tests for now * Fix linting issues * Make sure caching is off for tests * Remove unnecessary return * Use emulative parser if on PHP 5.6 * Cache parser for faster first-time parse * Fix constant resolution when scanning classes * Remove test that’s beyond a practical scope * Add back --diff support * Add --help for --threads * Remove unused vars
2017-07-25 16:11:02 -04:00
$threads,
$debug,
2018-06-04 10:19:20 -04:00
isset($options['report']) && is_string($options['report']) ? $options['report'] : null,
!isset($options['show-snippet']) || $options['show-snippet'] !== "false"
);
2017-01-16 11:05:29 -05:00
if (isset($options['php-version'])) {
if (!is_string($options['php-version'])) {
die('Expecting a version number in the format x.y' . PHP_EOL);
}
$project_analyzer->setPhpVersion($options['php-version']);
}
2018-11-11 12:01:14 -05:00
$project_analyzer->getCodebase()->diff_methods = isset($options['diff-methods']);
2019-02-24 01:33:25 -05:00
if ($type_map_location) {
$project_analyzer->getCodebase()->store_node_types = true;
}
2018-10-10 16:05:06 -04:00
$start_time = microtime(true);
2018-11-11 12:01:14 -05:00
$config->visitComposerAutoloadFiles($project_analyzer, $debug);
2018-10-10 16:05:06 -04:00
$now_time = microtime(true);
if ($debug) {
echo 'Visiting autoload files took ' . number_format($now_time - $start_time, 3) . 's' . "\n";
}
2018-03-26 09:08:55 -04:00
if (array_key_exists('debug-by-line', $options)) {
2018-11-11 12:01:14 -05:00
$project_analyzer->debug_lines = true;
2018-03-26 09:08:55 -04:00
}
2018-02-03 18:52:35 -05:00
if ($find_dead_code || $find_references_to !== null) {
2018-11-11 12:01:14 -05:00
$project_analyzer->getCodebase()->collectReferences();
if ($find_references_to) {
2018-11-11 12:01:14 -05:00
$project_analyzer->show_issues = false;
}
2018-02-03 18:52:35 -05:00
}
if ($find_dead_code) {
2018-11-11 12:01:14 -05:00
$project_analyzer->getCodebase()->reportUnusedCode();
}
/** @var string $plugin_path */
foreach ($plugins as $plugin_path) {
Config::getInstance()->addPluginPath($current_dir . $plugin_path);
}
if ($paths_to_check === null) {
2018-11-11 12:01:14 -05:00
$project_analyzer->check($current_dir, $is_diff);
} elseif ($paths_to_check) {
2018-11-11 12:01:14 -05:00
$project_analyzer->checkPaths($paths_to_check);
2016-07-26 15:12:44 -04:00
}
2018-01-21 23:42:57 -05:00
if ($find_references_to) {
2018-11-11 12:01:14 -05:00
$project_analyzer->findReferencesTo($find_references_to);
} elseif (($find_dead_code === 'always') || ($find_dead_code === 'auto' && !$paths_to_check && !$is_diff)) {
if ($threads > 1) {
2018-11-05 21:57:36 -05:00
if ($output_format === ProjectAnalyzer::TYPE_CONSOLE) {
echo 'Unused classes and methods cannot currently be found in multithreaded mode' . PHP_EOL;
}
} else {
2018-11-11 12:01:14 -05:00
$project_analyzer->checkClassReferences();
}
2018-01-21 23:42:57 -05:00
}
if (isset($options['set-baseline']) && is_string($options['set-baseline'])) {
echo 'Writing error baseline to file...', PHP_EOL;
ErrorBaseline::create(
2018-11-05 21:57:36 -05:00
new \Psalm\Internal\Provider\FileProvider,
$options['set-baseline'],
IssueBuffer::getIssuesData()
);
echo "Baseline saved to {$options['set-baseline']}.";
/** @var string $configFile */
$configFile = Config::locateConfigFile($path_to_config ?? $current_dir);
$configFileContents = $amendedConfigFileContents = file_get_contents($configFile);
if ($config->error_baseline) {
$amendedConfigFileContents = preg_replace(
'/errorBaseline=".*?"/',
"errorBaseline=\"{$options['set-baseline']}\"",
$configFileContents
);
} else {
$endPsalmOpenTag = strpos($configFileContents, '>', (int)strpos($configFileContents, '<psalm'));
if (!$endPsalmOpenTag) {
echo " Don't forget to set errorBaseline=\"{$options['set-baseline']}\" in your config.";
} elseif ($configFileContents[$endPsalmOpenTag - 1] === "\n") {
$amendedConfigFileContents = substr_replace(
$configFileContents,
" errorBaseline=\"{$options['set-baseline']}\"\n>",
$endPsalmOpenTag,
1
);
} else {
$amendedConfigFileContents = substr_replace(
$configFileContents,
" errorBaseline=\"{$options['set-baseline']}\">",
$endPsalmOpenTag,
1
);
}
}
file_put_contents($configFile, $amendedConfigFileContents);
echo PHP_EOL;
}
$issue_baseline = [];
if (isset($options['update-baseline'])) {
$baselineFile = Config::getInstance()->error_baseline;
if (empty($baselineFile)) {
die('Cannot update baseline, because no baseline file is configured.' . PHP_EOL);
}
try {
$issue_current_baseline = ErrorBaseline::read(
new \Psalm\Internal\Provider\FileProvider,
$baselineFile
);
$total_issues_current_baseline = ErrorBaseline::countTotalIssues($issue_current_baseline);
$issue_baseline = ErrorBaseline::update(
2018-11-05 21:57:36 -05:00
new \Psalm\Internal\Provider\FileProvider,
$baselineFile,
IssueBuffer::getIssuesData()
);
$total_issues_updated_baseline = ErrorBaseline::countTotalIssues($issue_baseline);
$total_fixed_issues = $total_issues_current_baseline - $total_issues_updated_baseline;
if ($total_fixed_issues > 0) {
echo str_repeat('-', 30) . "\n";
echo $total_fixed_issues . ' errors fixed' . "\n";
}
} catch (\Psalm\Exception\ConfigException $exception) {
die('Could not update baseline file: ' . $exception->getMessage());
}
}
if (!empty(Config::getInstance()->error_baseline) && !isset($options['ignore-baseline'])) {
try {
$issue_baseline = ErrorBaseline::read(
2018-11-05 21:57:36 -05:00
new \Psalm\Internal\Provider\FileProvider,
(string)Config::getInstance()->error_baseline
);
} catch (\Psalm\Exception\ConfigException $exception) {
die('Error while reading baseline: ' . $exception->getMessage());
}
}
2019-02-24 01:33:25 -05:00
if ($type_map_location) {
$file_map = $providers->file_reference_provider->getFileMaps();
$name_file_map = [];
$expected_references = [];
foreach ($file_map as $file_path => $map) {
$file_name = $config->shortenFileName($file_path);
foreach ($map[0] as [,$reference]) {
$expected_references[$reference] = true;
}
$map[2] = [];
$name_file_map[$file_name] = $map;
}
$reference_dictionary = [];
foreach ($providers->classlike_storage_provider->getAll() as $storage) {
if (!$storage->location) {
continue;
}
$fq_classlike_name = $storage->name;
if (isset($expected_references[$fq_classlike_name])) {
$reference_dictionary[$fq_classlike_name]
= $storage->location->file_name
. ':' . $storage->location->getLineNumber()
. ':' . $storage->location->getColumn();
}
foreach ($storage->methods as $method_name => $method_storage) {
if (!$method_storage->location) {
continue;
}
if (isset($expected_references[$fq_classlike_name . '::' . $method_name . '()'])) {
$reference_dictionary[$fq_classlike_name . '::' . $method_name . '()']
= $method_storage->location->file_name
. ':' . $method_storage->location->getLineNumber()
. ':' . $method_storage->location->getColumn();
}
}
foreach ($storage->properties as $property_name => $property_storage) {
if (!$property_storage->location) {
continue;
}
if (isset($expected_references[$fq_classlike_name . '::$' . $property_name])) {
$reference_dictionary[$fq_classlike_name . '::$' . $property_name]
= $property_storage->location->file_name
. ':' . $property_storage->location->getLineNumber()
. ':' . $property_storage->location->getColumn();
}
}
}
$providers->file_provider->setContents(
$type_map_location,
json_encode(['files' => $name_file_map, 'references' => $reference_dictionary])
);
}
IssueBuffer::finish(
2018-11-11 12:01:14 -05:00
$project_analyzer,
!$paths_to_check,
$start_time,
isset($options['stats']),
$issue_baseline
);