1
0
mirror of https://github.com/danog/psalm.git synced 2024-11-27 04:45:20 +01:00
psalm/src/psalter.php

210 lines
5.6 KiB
PHP
Raw Normal View History

<?php
require_once('command_functions.php');
use Psalm\Checker\ProjectChecker;
use Psalm\Config;
use Psalm\IssueBuffer;
// show all errors
error_reporting(-1);
ini_set('display_errors', 1);
ini_set('display_startup_errors', 1);
ini_set('memory_limit', '2048M');
// get options from command line
$options = getopt(
'f:mhr:',
[
'help', 'debug', 'config:', 'file:', 'root:',
2018-01-08 20:11:57 +01:00
'plugin:', 'issues:', 'php-version:', 'dry-run', 'safe-types',
]
);
if (array_key_exists('help', $options)) {
$options['h'] = false;
}
if (array_key_exists('monochrome', $options)) {
$options['m'] = false;
}
if (isset($options['config'])) {
$options['c'] = $options['config'];
}
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:
psalm [options] [file...]
Options:
-h, --help
Display this help message
--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
--plugin=PATH
Executes a plugin, an alternative to using the Psalm config
--dry-run
Shows a diff of all the changes, without making them
2018-01-07 23:17:18 +01:00
--safe-types
Only update PHP types when the new type information comes from other PHP types,
as opposed to type information that just comes from docblocks
--php-version=PHP_MAJOR_VERSION.PHP_MINOR_VERSION
--issues=IssueType1,IssueType2
If any issues can be fixed automatically, Psalm will update the codebase
HELP;
exit;
}
2018-01-07 17:48:33 +01:00
if (!isset($options['issues']) && (!isset($options['plugin']) || $options['plugin'] === false)) {
die('Please specify the issues you want to fix with --issues=IssueOne,IssueTwo '
. 'or provide a plugin that has its own manipulations with --plugin=path/to/plugin.php' . PHP_EOL);
}
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;
}
$vendor_dir = getVendorDir($current_dir);
$first_autoloader = requireAutoloaders($current_dir, isset($options['r']), $vendor_dir);
// If XDebug is enabled, restart without it
(new \Composer\XdebugHandler\XdebugHandler('PSALTER'))->check();
$paths_to_check = getPathsToCheck(isset($options['f']) ? $options['f'] : null);
if ($paths_to_check && count($paths_to_check) > 1) {
die('Psalter can currently only be run on one path at a time' . PHP_EOL);
}
$path_to_config = isset($options['c']) && is_string($options['c']) ? realpath($options['c']) : null;
if ($path_to_config === false) {
/** @psalm-suppress InvalidCast */
die('Could not resolve path to config ' . (string)$options['c'] . PHP_EOL);
}
// initialise custom config, if passed
if ($path_to_config) {
$config = Config::loadFromXMLFile($path_to_config, $current_dir);
} else {
$config = Config::getConfigForPath($current_dir, $current_dir, ProjectChecker::TYPE_CONSOLE);
}
$config->setComposerClassLoader($first_autoloader);
$project_checker = new ProjectChecker(
$config,
new Psalm\Provider\FileProvider(),
new Psalm\Provider\ParserCacheProvider(),
2018-02-19 06:27:39 +01:00
new Psalm\Provider\FileStorageCacheProvider($config),
new Psalm\Provider\ClassLikeStorageCacheProvider($config),
!array_key_exists('m', $options),
false,
ProjectChecker::TYPE_CONSOLE,
1,
array_key_exists('debug', $options)
);
$config->visitComposerAutoloadFiles($project_checker);
2018-01-07 17:48:33 +01:00
if (array_key_exists('issues', $options)) {
if (!is_string($options['issues']) || !$options['issues']) {
die('Expecting a comma-separated list of issues' . PHP_EOL);
}
$issues = explode(',', $options['issues']);
2018-01-07 17:48:33 +01:00
$keyed_issues = [];
2018-01-07 17:48:33 +01:00
foreach ($issues as $issue) {
$keyed_issues[$issue] = true;
}
} else {
$keyed_issues = [];
}
$php_major_version = PHP_MAJOR_VERSION;
$php_minor_version = PHP_MINOR_VERSION;
if (isset($options['php-version'])) {
2018-01-08 20:11:57 +01:00
if (!is_string($options['php-version']) || !preg_match('/^(5\.[456]|7\.[012])$/', $options['php-version'])) {
die('Expecting a version number in the format x.y' . PHP_EOL);
}
list($php_major_version, $php_minor_version) = explode('.', $options['php-version']);
}
2018-01-07 17:48:33 +01:00
$plugins = [];
if (isset($options['plugin'])) {
$plugins = $options['plugin'];
if (!is_array($plugins)) {
$plugins = [$plugins];
}
}
/** @var string $plugin_path */
foreach ($plugins as $plugin_path) {
Config::getInstance()->addPluginPath($current_dir . DIRECTORY_SEPARATOR . $plugin_path);
}
$project_checker->alterCodeAfterCompletion(
(int) $php_major_version,
(int) $php_minor_version,
2018-01-07 22:11:51 +01:00
array_key_exists('dry-run', $options),
array_key_exists('safe-types', $options)
2018-01-07 17:48:33 +01:00
);
$project_checker->setIssuesToFix($keyed_issues);
$start_time = (float) microtime(true);
if ($paths_to_check === null) {
$project_checker->check($current_dir);
} 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);
}
}
}
IssueBuffer::finish($project_checker, false, $start_time);