1
0
mirror of https://github.com/danog/psalm.git synced 2024-12-13 17:57:37 +01:00
psalm/src/psalm.php

412 lines
11 KiB
PHP
Raw Normal View History

2016-07-26 21:12:44 +02:00
<?php
2017-10-07 20:41:16 +02:00
use Psalm\Checker\ProjectChecker;
use Psalm\Config;
2016-07-26 21:12:44 +02:00
// show all errors
2016-10-30 17:46:18 +01:00
error_reporting(-1);
2016-07-26 21:12:44 +02:00
ini_set('display_errors', 1);
ini_set('display_startup_errors', 1);
ini_set('memory_limit', '2048M');
// get options from command line
2016-12-08 04:38:57 +01:00
$options = getopt(
'f:mhvc:ir:',
2016-12-08 04:38:57 +01:00
[
'help', 'debug', 'config:', 'monochrome', 'show-info:', 'diff',
'file:', 'self-check', 'update-docblocks', 'output-format:',
'find-dead-code', 'init', 'find-references-to:', 'root:', 'threads:',
'report:', 'clear-cache', 'no-cache', 'version', 'plugin:',
2016-12-08 04:38:57 +01:00
]
);
if (array_key_exists('help', $options)) {
2016-11-05 03:55:13 +01:00
$options['h'] = false;
}
if (array_key_exists('version', $options)) {
$options['v'] = false;
}
2017-02-13 05:59:33 +01:00
if (array_key_exists('init', $options)) {
$options['i'] = false;
}
if (array_key_exists('monochrome', $options)) {
2016-11-05 03:55:13 +01:00
$options['m'] = false;
}
if (isset($options['config'])) {
2016-11-05 03:55:13 +01:00
$options['c'] = $options['config'];
}
2017-01-17 01:44:31 +01:00
if (isset($options['c']) && is_array($options['c'])) {
die('Too many config files provided' . PHP_EOL);
}
if (array_key_exists('h', $options)) {
echo <<< HELP
Usage:
2016-11-05 03:55:13 +01:00
psalm [options] [file...]
Options:
2017-02-13 05:59:33 +01:00
-h, --help
Display this help message
-v, --version
Display the Psalm version
2017-02-13 05:59:33 +01:00
-i, --init [source_dir=src] [--level=3]
Create a psalm config file in the current directory that points to [source_dir]
at the required level, from 1, most strict, to 5, most permissive
--debug
Debug information
-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-13 05:59:33 +01:00
--show-info[=BOOLEAN]
Show non-exception parser findings
--diff
Runs Psalm in diff mode, only checking files that have changed (and their dependents)
--self-check
Psalm checks itself
--update-docblocks
Adds correct return types to the given file(s)
--output-format=console
2017-10-13 05:33:53 +02:00
Changes the output format. Possible values: console, json, xml
2017-02-13 05:59:33 +01:00
--find-dead-code
Look for dead code
--find-references-to=[class|method]
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 22:11:02 +02: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")
2017-10-07 20:41:16 +02:00
--clear-cache
Clears all cache files that Psalm uses
2017-10-14 03:27:20 +02:00
--no-cache
Runs Psalm without using cache
HELP;
exit;
}
2016-07-26 21:12:44 +02:00
if (getcwd() === false) {
die('Cannot get current working directory');
}
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) {
die('Could not locate root directory ' . $current_dir . DIRECTORY_SEPARATOR . $options['r'] . PHP_EOL);
}
$current_dir = $root_path . DIRECTORY_SEPARATOR;
}
$autoload_roots = [$current_dir];
$psalm_dir = dirname(__DIR__);
if (realpath($psalm_dir) !== realpath($current_dir)) {
$autoload_roots[] = $psalm_dir;
}
$autoload_files = [];
foreach ($autoload_roots as $autoload_root) {
$has_autoloader = false;
$nested_autoload_file = dirname(dirname($autoload_root)) . DIRECTORY_SEPARATOR . 'autoload.php';
if (file_exists($nested_autoload_file)) {
$autoload_files[] = realpath($nested_autoload_file);
$has_autoloader = true;
}
$vendor_autoload_file = $autoload_root . DIRECTORY_SEPARATOR . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php';
if (file_exists($vendor_autoload_file)) {
$autoload_files[] = realpath($vendor_autoload_file);
$has_autoloader = true;
}
if (!$has_autoloader) {
$error_message = 'Could not find any composer autoloaders in ' . $autoload_root;
if (!isset($options['r'])) {
$error_message .=
PHP_EOL . 'Add a --root=[your/project/directory] flag to specify a particular project to run Psalm on.';
}
die($error_message . PHP_EOL);
}
}
foreach ($autoload_files as $file) {
/** @psalm-suppress UnresolvableInclude */
require_once $file;
}
if (array_key_exists('v', $options)) {
/** @var string */
2018-01-01 16:01:04 +01:00
$version = \Muglug\PackageVersions\Versions::getVersion('vimeo/psalm');
echo 'Psalm ' . $version . PHP_EOL;
exit;
}
2017-10-07 20:41:16 +02:00
// If XDebug is enabled, restart without it
(new \Composer\XdebugHandler(\Composer\Factory::createOutput()))->check();
2017-02-13 05:59:33 +01:00
if (isset($options['i'])) {
if (file_exists('psalm.xml')) {
die('A config file already exists in the current directory' . PHP_EOL);
}
$args = array_values(array_filter(
array_slice($argv, 2),
/**
* @param string $arg
*
* @return bool
*/
function ($arg) {
return $arg !== '--ansi' && $arg !== '--no-ansi';
}
));
2017-02-13 05:59:33 +01: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-5]$/', $args[1])) {
die('Config strictness must be a number between 1 and 5 inclusive' . PHP_EOL);
}
$level = (int)$args[1];
}
2017-02-13 06:12:56 +01: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-13 05:59:33 +01:00
}
2017-02-13 06:12:56 +01:00
die('The given path "' . $bad_dir_path . '" does not appear to be a directory' . PHP_EOL);
2017-02-13 05:59:33 +01:00
}
2017-05-28 06:07:26 +02:00
$template_file_name = dirname(__DIR__) . '/assets/config_levels/' . $level . '.xml';
2017-02-13 05:59:33 +01:00
if (!file_exists($template_file_name)) {
2017-05-28 06:07:26 +02:00
die('Could not open config template ' . $template_file_name . PHP_EOL);
2017-02-13 05:59:33 +01:00
}
$template = (string)file_get_contents($template_file_name);
$template = str_replace('<projectFiles>
<directory name="src" />
</projectFiles>', '<projectFiles>
<directory name="' . $source_dir . '" />
</projectFiles>', $template);
if (!file_put_contents('psalm.xml', $template)) {
die('Could not write to psalm.xml' . PHP_EOL);
}
exit('Config file created successfully. Please re-run psalm.' . PHP_EOL);
}
2016-07-26 21:12:44 +02:00
// get vars from options
$debug = array_key_exists('debug', $options);
if (isset($options['f'])) {
2016-11-05 03:55:13 +01:00
$input_paths = is_array($options['f']) ? $options['f'] : [$options['f']];
2016-12-08 04:38:57 +01:00
} else {
2016-11-05 03:55:13 +01:00
$input_paths = $argv ? $argv : null;
}
$output_format = isset($options['output-format']) && is_string($options['output-format'])
? $options['output-format']
: ProjectChecker::TYPE_CONSOLE;
2016-12-08 04:38:57 +01:00
2016-11-05 03:55:13 +01:00
$paths_to_check = null;
if ($input_paths) {
2016-11-05 06:50:45 +01:00
$filtered_input_paths = [];
for ($i = 0; $i < count($input_paths); ++$i) {
/** @var string */
2017-01-16 19:46:59 +01:00
$input_path = $input_paths[$i];
if (realpath($input_path) === realpath(dirname(__DIR__) . DIRECTORY_SEPARATOR . 'psalm')) {
2016-11-05 06:50:45 +01:00
continue;
}
2017-01-16 19:46:59 +01:00
if ($input_path[0] === '-' && strlen($input_path) === 2) {
2017-01-17 01:44:31 +01:00
if ($input_path[1] === 'c' || $input_path[1] === 'f') {
++$i;
2017-01-17 01:44:31 +01:00
}
2017-01-16 19:46:59 +01:00
continue;
}
if ($input_path[0] === '-' && $input_path[2] === '=') {
continue;
}
if (substr($input_path, 0, 2) === '--' && strlen($input_path) > 2) {
continue;
}
$filtered_input_paths[] = $input_path;
2016-11-05 06:50:45 +01:00
}
stream_set_blocking(STDIN, false);
2016-11-05 06:50:45 +01:00
if ($filtered_input_paths === ['-'] && $stdin = fgets(STDIN)) {
$filtered_input_paths = preg_split('/\s+/', trim($stdin));
}
foreach ($filtered_input_paths as $i => $path_to_check) {
2016-11-05 03:55:13 +01:00
if ($path_to_check[0] === '-') {
2016-11-05 06:50:45 +01:00
die('Invalid usage, expecting psalm [options] [file...]' . PHP_EOL);
2016-11-05 03:55:13 +01:00
}
if (!file_exists($path_to_check)) {
die('Cannot locate ' . $path_to_check . PHP_EOL);
}
$path_to_check = realpath($path_to_check);
if (!$path_to_check) {
die('Error getting realpath for file' . PHP_EOL);
}
$paths_to_check[] = $path_to_check;
2016-11-05 03:55:13 +01:00
}
if (!$paths_to_check) {
$paths_to_check = null;
}
}
$path_to_config = isset($options['c']) && is_string($options['c']) ? realpath($options['c']) : null;
2016-11-05 03:55:13 +01:00
if ($path_to_config === false) {
/** @psalm-suppress InvalidCast */
die('Could not resolve path to config ' . (string)$options['c'] . PHP_EOL);
}
$use_color = !array_key_exists('m', $options);
2016-11-05 03:55:13 +01:00
$show_info = isset($options['show-info'])
? $options['show-info'] !== 'false' && $options['show-info'] !== '0'
: true;
2016-11-05 03:55:13 +01:00
2016-10-07 06:58:08 +02:00
$is_diff = isset($options['diff']);
2016-07-26 21:12:44 +02:00
$find_dead_code = isset($options['find-dead-code']);
$find_references_to = isset($options['find-references-to']) && is_string($options['find-references-to'])
? $options['find-references-to']
: null;
2016-11-13 00:51:48 +01:00
$update_docblocks = isset($options['update-docblocks']);
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 22:11:02 +02:00
$threads = isset($options['threads']) ? (int)$options['threads'] : 1;
2017-10-14 03:27:20 +02:00
$cache_provider = isset($options['no-cache'])
2017-10-15 17:57:44 +02:00
? new Psalm\Provider\Cache\NoParserCacheProvider()
: new Psalm\Provider\ParserCacheProvider();
2017-10-14 03:27:20 +02:00
$project_checker = new ProjectChecker(
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 22:11:02 +02:00
new Psalm\Provider\FileProvider(),
2017-10-14 03:27:20 +02:00
$cache_provider,
$use_color,
$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 22:11:02 +02:00
$threads,
$debug,
$update_docblocks,
$find_dead_code || $find_references_to !== null,
$find_references_to,
isset($options['report']) && is_string($options['report']) ? $options['report'] : null
);
2017-01-16 17:05:29 +01:00
2016-07-26 21:12:44 +02:00
// initialise custom config, if passed
if ($path_to_config) {
$project_checker->setConfigXML($path_to_config, $current_dir);
2016-07-26 21:12:44 +02:00
}
2017-10-07 20:41:16 +02:00
if (isset($options['clear-cache'])) {
// initialise config if it hasn't already been
if (!$path_to_config) {
$project_checker->getConfigForPath($current_dir, $current_dir);
2017-10-07 20:41:16 +02:00
}
$cache_directory = Config::getInstance()->getCacheDirectory();
Config::removeCacheDirectory($cache_directory);
2017-10-07 20:41:16 +02:00
echo 'Cache directory deleted' . PHP_EOL;
exit;
}
2017-02-18 23:49:05 +01:00
$config = $project_checker->getConfig();
if (!$config) {
$project_checker->getConfigForPath($current_dir, $current_dir);
}
/** @psalm-suppress MixedArgument */
2016-12-15 01:24:16 +01:00
\Psalm\IssueBuffer::setStartTime(microtime(true));
2016-07-26 21:12:44 +02:00
2016-11-05 01:24:43 +01:00
if (array_key_exists('self-check', $options)) {
$project_checker->checkDir(__DIR__);
2016-11-05 01:24:43 +01:00
} elseif ($paths_to_check === null) {
$project_checker->check($current_dir, $is_diff);
} elseif ($paths_to_check) {
foreach ($paths_to_check as $path_to_check) {
if (is_dir($path_to_check)) {
$project_checker->checkDir($path_to_check);
} else {
$project_checker->checkFile($path_to_check);
2016-07-26 21:12:44 +02:00
}
}
}