2016-07-26 21:12:44 +02:00
|
|
|
<?php
|
2018-01-07 06:11:23 +01:00
|
|
|
require_once('command_functions.php');
|
2016-07-26 21:12:44 +02:00
|
|
|
|
2017-10-07 20:41:16 +02:00
|
|
|
use Psalm\Checker\ProjectChecker;
|
|
|
|
use Psalm\Config;
|
2018-01-09 17:49:10 +01:00
|
|
|
use Psalm\IssueBuffer;
|
2017-10-07 20:41:16 +02:00
|
|
|
|
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(
|
2017-11-11 18:11:11 +01:00
|
|
|
'f:mhvc:ir:',
|
2016-12-08 04:38:57 +01:00
|
|
|
[
|
2018-03-26 15:08:55 +02:00
|
|
|
'help', 'debug', 'debug-by-line', 'config:', 'monochrome', 'show-info:', 'diff',
|
2018-03-05 16:31:07 +01:00
|
|
|
'output-format:', 'report:', 'find-dead-code', 'init',
|
2018-01-07 06:11:23 +01:00
|
|
|
'find-references-to:', 'root:', 'threads:', 'clear-cache', 'no-cache',
|
2018-01-31 23:09:09 +01:00
|
|
|
'version', 'plugin:', 'stats',
|
2016-12-08 04:38:57 +01:00
|
|
|
]
|
|
|
|
);
|
2016-10-18 23:00:03 +02:00
|
|
|
|
2016-11-05 01:13:16 +01:00
|
|
|
if (array_key_exists('help', $options)) {
|
2016-11-05 03:55:13 +01:00
|
|
|
$options['h'] = false;
|
2016-11-05 01:13:16 +01:00
|
|
|
}
|
|
|
|
|
2017-11-11 18:11:11 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2016-11-05 01:13:16 +01:00
|
|
|
if (array_key_exists('monochrome', $options)) {
|
2016-11-05 03:55:13 +01:00
|
|
|
$options['m'] = false;
|
2016-11-05 01:13:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($options['config'])) {
|
2016-11-05 03:55:13 +01:00
|
|
|
$options['c'] = $options['config'];
|
2016-11-05 01:13:16 +01:00
|
|
|
}
|
|
|
|
|
2017-01-17 01:44:31 +01:00
|
|
|
if (isset($options['c']) && is_array($options['c'])) {
|
2018-04-17 17:48:29 +02:00
|
|
|
echo 'Too many config files provided' . PHP_EOL;
|
|
|
|
exit(1);
|
2017-01-17 01:44:31 +01:00
|
|
|
}
|
|
|
|
|
2016-11-05 01:13:16 +01:00
|
|
|
if (array_key_exists('h', $options)) {
|
2017-11-28 22:29:51 +01:00
|
|
|
echo <<<HELP
|
2016-10-18 23:00:03 +02:00
|
|
|
Usage:
|
2016-11-05 03:55:13 +01:00
|
|
|
psalm [options] [file...]
|
2016-10-18 23:00:03 +02:00
|
|
|
|
|
|
|
Options:
|
2017-02-13 05:59:33 +01:00
|
|
|
-h, --help
|
|
|
|
Display this help message
|
|
|
|
|
2017-11-11 18:11:11 +01:00
|
|
|
-v, --version
|
|
|
|
Display the Psalm version
|
|
|
|
|
2018-03-02 16:58:12 +01:00
|
|
|
-i, --init [source_dir=src] [level=3]
|
2017-02-13 05:59:33 +01:00
|
|
|
Create a psalm config file in the current directory that points to [source_dir]
|
2018-04-19 07:03:28 +02:00
|
|
|
at the required level, from 1, most strict, to 8, most permissive.
|
2017-02-13 05:59:33 +01:00
|
|
|
|
|
|
|
--debug
|
|
|
|
Debug information
|
|
|
|
|
2018-03-26 15:08:55 +02:00
|
|
|
--debug-by-line
|
|
|
|
Debug information on a line-by-line level
|
|
|
|
|
2017-02-13 05:59:33 +01:00
|
|
|
-c, --config=psalm.xml
|
|
|
|
Path to a psalm.xml configuration file. Run psalm --init to create one.
|
|
|
|
|
|
|
|
-m, --monochrome
|
|
|
|
Enable monochrome output
|
|
|
|
|
2017-05-04 20:25:58 +02:00
|
|
|
-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)
|
|
|
|
|
|
|
|
--output-format=console
|
2018-02-19 06:57:06 +01:00
|
|
|
Changes the output format. Possible values: console, emacs, json, pylint, xml
|
2017-02-13 05:59:33 +01:00
|
|
|
|
|
|
|
--find-dead-code
|
|
|
|
Look for dead code
|
2016-10-18 23:00:03 +02:00
|
|
|
|
2017-02-27 07:30:44 +01:00
|
|
|
--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
|
2016-10-18 23:00:03 +02:00
|
|
|
|
2017-07-25 22:11:02 +02:00
|
|
|
--threads=INT
|
|
|
|
If greater than one, Psalm will run analysis on multiple threads, speeding things up.
|
|
|
|
|
2017-09-08 17:18:48 +02:00
|
|
|
--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
|
|
|
|
|
2018-01-02 03:17:23 +01:00
|
|
|
--plugin=PATH
|
|
|
|
Executes a plugin, an alternative to using the Psalm config
|
|
|
|
|
2018-01-31 23:09:09 +01:00
|
|
|
--stats
|
|
|
|
Shows a breakdown of Psalm's ability to infer types in the codebase
|
|
|
|
|
2016-10-18 23:00:03 +02:00
|
|
|
HELP;
|
|
|
|
|
|
|
|
exit;
|
|
|
|
}
|
2016-07-26 21:12:44 +02:00
|
|
|
|
2017-01-17 00:33:04 +01:00
|
|
|
if (getcwd() === false) {
|
2018-04-17 17:48:29 +02:00
|
|
|
echo 'Cannot get current working directory' . PHP_EOL;
|
|
|
|
exit(1);
|
2017-01-17 00:33:04 +01:00
|
|
|
}
|
|
|
|
|
2017-05-04 20:25:58 +02:00
|
|
|
if (isset($options['root'])) {
|
|
|
|
$options['r'] = $options['root'];
|
|
|
|
}
|
|
|
|
|
|
|
|
$current_dir = (string)getcwd() . DIRECTORY_SEPARATOR;
|
|
|
|
|
2017-12-29 18:29:36 +01:00
|
|
|
if (isset($options['r']) && is_string($options['r'])) {
|
2017-05-04 20:25:58 +02:00
|
|
|
$root_path = realpath($options['r']);
|
|
|
|
|
|
|
|
if (!$root_path) {
|
2018-04-17 17:48:29 +02:00
|
|
|
echo 'Could not locate root directory ' . $current_dir . DIRECTORY_SEPARATOR . $options['r'] . PHP_EOL;
|
|
|
|
exit(1);
|
2017-05-04 20:25:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$current_dir = $root_path . DIRECTORY_SEPARATOR;
|
|
|
|
}
|
|
|
|
|
2018-01-25 17:32:54 +01:00
|
|
|
$vendor_dir = getVendorDir($current_dir);
|
|
|
|
|
2018-03-03 21:19:05 +01:00
|
|
|
$first_autoloader = requireAutoloaders($current_dir, isset($options['r']), $vendor_dir);
|
2017-05-04 20:25:58 +02:00
|
|
|
|
2017-11-11 18:11:11 +01:00
|
|
|
if (array_key_exists('v', $options)) {
|
2018-02-25 17:35:43 +01:00
|
|
|
echo 'Psalm ' . PSALM_VERSION . PHP_EOL;
|
2017-11-11 18:11:11 +01:00
|
|
|
exit;
|
|
|
|
}
|
|
|
|
|
2017-10-07 20:41:16 +02:00
|
|
|
// If XDebug is enabled, restart without it
|
2018-03-05 08:58:25 +01:00
|
|
|
(new \Composer\XdebugHandler\XdebugHandler('PSALM'))->check();
|
2017-10-07 20:41:16 +02:00
|
|
|
|
2018-02-18 04:11:42 +01:00
|
|
|
setlocale(LC_CTYPE, 'C');
|
|
|
|
|
2017-02-13 05:59:33 +01:00
|
|
|
if (isset($options['i'])) {
|
2018-03-19 00:16:35 +01:00
|
|
|
if (file_exists($current_dir . 'psalm.xml')) {
|
2017-02-13 05:59:33 +01:00
|
|
|
die('A config file already exists in the current directory' . PHP_EOL);
|
|
|
|
}
|
|
|
|
|
2017-11-10 22:44:16 +01:00
|
|
|
$args = array_values(array_filter(
|
2018-03-19 00:16:35 +01:00
|
|
|
array_slice($argv, 1),
|
2017-12-29 18:29:36 +01:00
|
|
|
/**
|
|
|
|
* @param string $arg
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2017-10-14 03:16:34 +02:00
|
|
|
function ($arg) {
|
2018-03-19 00:16:35 +01:00
|
|
|
return $arg !== '--ansi'
|
|
|
|
&& $arg !== '--no-ansi'
|
|
|
|
&& $arg !== '--i'
|
|
|
|
&& $arg !== '--init'
|
|
|
|
&& strpos($arg, '--root=') !== 0
|
|
|
|
&& strpos($arg, '--r=') !== 0;
|
2017-10-14 03:16:34 +02:00
|
|
|
}
|
2017-11-10 22:44:16 +01:00
|
|
|
));
|
2017-10-14 03:16:34 +02:00
|
|
|
|
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])) {
|
2018-04-19 07:03:28 +02:00
|
|
|
if (!preg_match('/^[1-8]$/', $args[1])) {
|
|
|
|
die('Config strictness must be a number between 1 and 8 inclusive' . PHP_EOL);
|
2017-02-13 05:59:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$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);
|
|
|
|
|
2018-03-18 03:33:25 +01: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'),
|
2018-03-18 03:33:25 +01:00
|
|
|
$template
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-03-19 00:16:35 +01:00
|
|
|
if (!file_put_contents($current_dir . 'psalm.xml', $template)) {
|
2017-02-13 05:59:33 +01:00
|
|
|
die('Could not write to psalm.xml' . PHP_EOL);
|
|
|
|
}
|
|
|
|
|
|
|
|
exit('Config file created successfully. Please re-run psalm.' . PHP_EOL);
|
|
|
|
}
|
|
|
|
|
2017-12-29 18:29:36 +01:00
|
|
|
$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
|
|
|
|
2018-01-07 06:11:23 +01:00
|
|
|
$paths_to_check = getPathsToCheck(isset($options['f']) ? $options['f'] : null);
|
2016-11-05 01:13:16 +01:00
|
|
|
|
2018-01-02 02:05:54 +01:00
|
|
|
$plugins = [];
|
|
|
|
|
|
|
|
if (isset($options['plugin'])) {
|
|
|
|
$plugins = $options['plugin'];
|
|
|
|
|
|
|
|
if (!is_array($plugins)) {
|
|
|
|
$plugins = [$plugins];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-29 18:29:36 +01:00
|
|
|
$path_to_config = isset($options['c']) && is_string($options['c']) ? realpath($options['c']) : null;
|
2016-11-05 03:55:13 +01:00
|
|
|
|
2017-01-16 17:41:57 +01:00
|
|
|
if ($path_to_config === false) {
|
2017-12-29 18:29:36 +01:00
|
|
|
/** @psalm-suppress InvalidCast */
|
2018-04-17 17:48:29 +02:00
|
|
|
echo 'Could not resolve path to config ' . (string)$options['c'] . PHP_EOL;
|
|
|
|
exit(1);
|
2017-01-16 17:41:57 +01:00
|
|
|
}
|
|
|
|
|
2016-08-04 20:38:43 +02:00
|
|
|
$show_info = isset($options['show-info'])
|
2018-01-07 06:11:23 +01:00
|
|
|
? $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
|
|
|
|
2017-02-01 02:47:16 +01:00
|
|
|
$find_dead_code = isset($options['find-dead-code']);
|
|
|
|
|
2017-12-29 18:29:36 +01:00
|
|
|
$find_references_to = isset($options['find-references-to']) && is_string($options['find-references-to'])
|
|
|
|
? $options['find-references-to']
|
|
|
|
: null;
|
2017-02-27 07:30:44 +01:00
|
|
|
|
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'])
|
2018-02-19 06:27:39 +01:00
|
|
|
? new Psalm\Provider\NoCache\NoParserCacheProvider()
|
2017-10-15 17:57:44 +02:00
|
|
|
: new Psalm\Provider\ParserCacheProvider();
|
2017-10-14 03:27:20 +02:00
|
|
|
|
2018-01-21 16:22:04 +01:00
|
|
|
// initialise custom config, if passed
|
2018-04-19 23:29:07 +02: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);
|
2018-01-21 16:22:04 +01:00
|
|
|
}
|
|
|
|
|
2018-03-03 21:19:05 +01:00
|
|
|
$config->setComposerClassLoader($first_autoloader);
|
|
|
|
|
2018-02-19 17:14:07 +01:00
|
|
|
$file_storage_cache_provider = isset($options['no-cache'])
|
|
|
|
? new Psalm\Provider\NoCache\NoFileStorageCacheProvider()
|
|
|
|
: new Psalm\Provider\FileStorageCacheProvider($config);
|
2018-02-19 06:27:39 +01:00
|
|
|
|
2018-02-19 17:14:07 +01:00
|
|
|
$classlike_storage_cache_provider = isset($options['no-cache'])
|
|
|
|
? new Psalm\Provider\NoCache\NoClassLikeStorageCacheProvider()
|
|
|
|
: new Psalm\Provider\ClassLikeStorageCacheProvider($config);
|
2018-02-19 06:27:39 +01:00
|
|
|
|
2018-02-08 02:15:56 +01:00
|
|
|
if (isset($options['clear-cache'])) {
|
|
|
|
$cache_directory = $config->getCacheDirectory();
|
|
|
|
|
|
|
|
Config::removeCacheDirectory($cache_directory);
|
|
|
|
echo 'Cache directory deleted' . PHP_EOL;
|
|
|
|
exit;
|
|
|
|
}
|
|
|
|
|
2017-02-01 02:47:16 +01:00
|
|
|
$project_checker = new ProjectChecker(
|
2018-01-21 16:22:04 +01:00
|
|
|
$config,
|
2017-07-25 22:11:02 +02:00
|
|
|
new Psalm\Provider\FileProvider(),
|
2017-10-14 03:27:20 +02:00
|
|
|
$cache_provider,
|
2018-02-19 06:27:39 +01:00
|
|
|
$file_storage_cache_provider,
|
|
|
|
$classlike_storage_cache_provider,
|
2018-01-07 06:11:23 +01:00
|
|
|
!array_key_exists('m', $options),
|
2017-02-01 02:47:16 +01:00
|
|
|
$show_info,
|
|
|
|
$output_format,
|
2017-07-25 22:11:02 +02:00
|
|
|
$threads,
|
2018-03-26 15:08:55 +02:00
|
|
|
array_key_exists('debug', $options) || array_key_exists('debug-by-line', $options),
|
2017-12-29 18:29:36 +01:00
|
|
|
isset($options['report']) && is_string($options['report']) ? $options['report'] : null
|
2017-02-01 02:47:16 +01:00
|
|
|
);
|
2017-01-16 17:05:29 +01:00
|
|
|
|
2018-03-31 02:03:56 +02:00
|
|
|
$config->visitComposerAutoloadFiles($project_checker);
|
|
|
|
|
2018-03-26 15:08:55 +02:00
|
|
|
if (array_key_exists('debug-by-line', $options)) {
|
|
|
|
$project_checker->debug_lines = true;
|
|
|
|
}
|
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
if ($find_dead_code || $find_references_to !== null) {
|
|
|
|
$project_checker->getCodebase()->collectReferences();
|
2018-03-16 16:15:01 +01:00
|
|
|
|
|
|
|
if ($find_references_to) {
|
|
|
|
$project_checker->show_issues = false;
|
|
|
|
}
|
2018-02-04 00:52:35 +01:00
|
|
|
}
|
|
|
|
|
2018-02-17 23:45:30 +01:00
|
|
|
if ($find_dead_code) {
|
|
|
|
$project_checker->getCodebase()->reportUnusedCode();
|
|
|
|
}
|
|
|
|
|
2018-01-02 02:05:54 +01:00
|
|
|
/** @var string $plugin_path */
|
|
|
|
foreach ($plugins as $plugin_path) {
|
|
|
|
Config::getInstance()->addPluginPath($current_dir . DIRECTORY_SEPARATOR . $plugin_path);
|
|
|
|
}
|
|
|
|
|
2018-01-09 17:49:10 +01:00
|
|
|
$start_time = (float) microtime(true);
|
|
|
|
|
2018-03-05 16:31:07 +01:00
|
|
|
if ($paths_to_check === null) {
|
2017-05-04 20:25:58 +02:00
|
|
|
$project_checker->check($current_dir, $is_diff);
|
2016-11-05 01:13:16 +01:00
|
|
|
} elseif ($paths_to_check) {
|
|
|
|
foreach ($paths_to_check as $path_to_check) {
|
|
|
|
if (is_dir($path_to_check)) {
|
2018-01-02 02:04:03 +01:00
|
|
|
$project_checker->checkDir($path_to_check);
|
2016-11-05 01:13:16 +01:00
|
|
|
} else {
|
2018-01-02 02:04:03 +01:00
|
|
|
$project_checker->checkFile($path_to_check);
|
2016-07-26 21:12:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-01-09 17:49:10 +01:00
|
|
|
|
2018-01-22 05:42:57 +01:00
|
|
|
if ($find_references_to) {
|
|
|
|
$project_checker->findReferencesTo($find_references_to);
|
2018-01-25 21:11:44 +01:00
|
|
|
} elseif ($find_dead_code && !$paths_to_check && !$is_diff) {
|
2018-01-25 21:36:02 +01:00
|
|
|
if ($threads > 1) {
|
|
|
|
if ($output_format === ProjectChecker::TYPE_CONSOLE) {
|
2018-01-25 21:40:01 +01:00
|
|
|
echo 'Unused classes and methods cannot currently be found in multithreaded mode' . PHP_EOL;
|
2018-01-25 21:36:02 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$project_checker->checkClassReferences();
|
|
|
|
}
|
2018-01-22 05:42:57 +01:00
|
|
|
}
|
|
|
|
|
2018-02-20 05:36:29 +01:00
|
|
|
IssueBuffer::finish($project_checker, !$paths_to_check, $start_time, isset($options['stats']));
|