mirror of
https://github.com/danog/psalm.git
synced 2025-01-10 15:09:04 +01:00
338 lines
10 KiB
PHP
338 lines
10 KiB
PHP
<?php
|
|
|
|
namespace Psalm\Internal\Cli;
|
|
|
|
use AssertionError;
|
|
use Composer\XdebugHandler\XdebugHandler;
|
|
use Psalm\Internal\Analyzer\ProjectAnalyzer;
|
|
use Psalm\Internal\CliUtils;
|
|
use Psalm\Internal\Composer;
|
|
use Psalm\Internal\ErrorHandler;
|
|
use Psalm\Internal\IncludeCollector;
|
|
use Psalm\Internal\Provider\ClassLikeStorageCacheProvider;
|
|
use Psalm\Internal\Provider\FileProvider;
|
|
use Psalm\Internal\Provider\FileStorageCacheProvider;
|
|
use Psalm\Internal\Provider\ParserCacheProvider;
|
|
use Psalm\Internal\Provider\ProjectCacheProvider;
|
|
use Psalm\Internal\Provider\Providers;
|
|
use Psalm\IssueBuffer;
|
|
use Psalm\Progress\DebugProgress;
|
|
use Psalm\Progress\DefaultProgress;
|
|
use Psalm\Report;
|
|
use Psalm\Report\ReportOptions;
|
|
|
|
use function array_key_exists;
|
|
use function array_map;
|
|
use function array_slice;
|
|
use function chdir;
|
|
use function end;
|
|
use function explode;
|
|
use function fwrite;
|
|
use function gc_collect_cycles;
|
|
use function gc_disable;
|
|
use function getcwd;
|
|
use function getopt;
|
|
use function implode;
|
|
use function in_array;
|
|
use function ini_set;
|
|
use function is_array;
|
|
use function is_string;
|
|
use function max;
|
|
use function microtime;
|
|
use function preg_last_error_msg;
|
|
use function preg_replace;
|
|
use function preg_split;
|
|
use function realpath;
|
|
use function strpos;
|
|
use function substr;
|
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
use const PHP_EOL;
|
|
use const STDERR;
|
|
|
|
// phpcs:disable PSR1.Files.SideEffects
|
|
|
|
require_once __DIR__ . '/../ErrorHandler.php';
|
|
require_once __DIR__ . '/../CliUtils.php';
|
|
require_once __DIR__ . '/../Composer.php';
|
|
require_once __DIR__ . '/../IncludeCollector.php';
|
|
require_once __DIR__ . '/../../IssueBuffer.php';
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
final class Refactor
|
|
{
|
|
/** @param array<int,string> $argv */
|
|
public static function run(array $argv): void
|
|
{
|
|
ini_set('memory_limit', '8192M');
|
|
|
|
gc_collect_cycles();
|
|
gc_disable();
|
|
|
|
ErrorHandler::install();
|
|
|
|
$args = array_slice($argv, 1);
|
|
|
|
$valid_short_options = ['f:', 'm', 'h', 'r:', 'c:'];
|
|
$valid_long_options = [
|
|
'help', 'debug', 'debug-by-line', 'debug-emitted-issues', 'config:', 'root:',
|
|
'threads:', 'move:', 'into:', 'rename:', 'to:',
|
|
];
|
|
|
|
// get options from command line
|
|
$options = getopt(implode('', $valid_short_options), $valid_long_options);
|
|
|
|
array_map(
|
|
static function (string $arg) use ($valid_long_options): void {
|
|
if (strpos($arg, '--') === 0 && $arg !== '--') {
|
|
$arg_name = preg_replace('/=.*$/', '', substr($arg, 2));
|
|
|
|
if ($arg_name === 'refactor') {
|
|
// valid option for psalm, ignored by psalter
|
|
return;
|
|
}
|
|
|
|
if (!in_array($arg_name, $valid_long_options)
|
|
&& !in_array($arg_name . ':', $valid_long_options)
|
|
&& !in_array($arg_name . '::', $valid_long_options)
|
|
) {
|
|
fwrite(
|
|
STDERR,
|
|
'Unrecognised argument "--' . $arg_name . '"' . PHP_EOL
|
|
. 'Type --help to see a list of supported arguments'. PHP_EOL
|
|
);
|
|
exit(1);
|
|
}
|
|
}
|
|
},
|
|
$args
|
|
);
|
|
|
|
if (array_key_exists('help', $options)) {
|
|
$options['h'] = 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-refactor [options] [symbol1] into [symbol2]
|
|
|
|
Options:
|
|
-h, --help
|
|
Display this help message
|
|
|
|
--debug, --debug-by-line, --debug-emitted-issues
|
|
Debug information
|
|
|
|
-c, --config=psalm.xml
|
|
Path to a psalm.xml configuration file. Run psalm --init to create one.
|
|
|
|
-r, --root
|
|
If running Psalm globally you'll need to specify a project root. Defaults to cwd
|
|
|
|
--threads=auto
|
|
If greater than one, Psalm will run analysis on multiple threads, speeding things up.
|
|
By default
|
|
|
|
--move "[Identifier]" --into "[Class]"
|
|
Moves the specified item into the class. More than one item can be moved into a class
|
|
by passing a comma-separated list of values e.g.
|
|
|
|
--move "Ns\Foo::bar,Ns\Foo::baz" --into "Biz\Bang\DestinationClass"
|
|
|
|
--rename "[Identifier]" --to "[NewIdentifier]"
|
|
Renames a specified item (e.g. method) and updates all references to it that Psalm can
|
|
identify.
|
|
|
|
HELP;
|
|
|
|
exit;
|
|
}
|
|
|
|
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 = CliUtils::getVendorDir($current_dir);
|
|
|
|
// capture environment before registering autoloader (it may destroy it)
|
|
IssueBuffer::captureServer($_SERVER);
|
|
|
|
$include_collector = new IncludeCollector();
|
|
$first_autoloader = $include_collector->runAndCollect(
|
|
// we ignore the FQN because of a hack in scoper.inc that needs full path
|
|
// phpcs:ignore SlevomatCodingStandard.Namespaces.ReferenceUsedNamesOnly.ReferenceViaFullyQualifiedName
|
|
static fn(): ?\Composer\Autoload\ClassLoader =>
|
|
CliUtils::requireAutoloaders($current_dir, isset($options['r']), $vendor_dir)
|
|
);
|
|
|
|
// If Xdebug is enabled, restart without it
|
|
(new XdebugHandler('PSALTER'))->check();
|
|
|
|
$path_to_config = CliUtils::getPathToConfig($options);
|
|
|
|
$args = CliUtils::getArguments();
|
|
|
|
$operation = null;
|
|
$last_arg = null;
|
|
|
|
$to_refactor = [];
|
|
|
|
foreach ($args as $arg) {
|
|
if ($arg === '--move') {
|
|
$operation = 'move';
|
|
continue;
|
|
}
|
|
|
|
if ($arg === '--into') {
|
|
if ($operation !== 'move' || !$last_arg) {
|
|
die('--into is not expected here' . PHP_EOL);
|
|
}
|
|
|
|
$operation = 'move_into';
|
|
continue;
|
|
}
|
|
|
|
if ($arg === '--rename') {
|
|
$operation = 'rename';
|
|
continue;
|
|
}
|
|
|
|
if ($arg === '--to') {
|
|
if ($operation !== 'rename' || !$last_arg) {
|
|
die('--to is not expected here' . PHP_EOL);
|
|
}
|
|
|
|
$operation = 'rename_to';
|
|
|
|
continue;
|
|
}
|
|
|
|
if ($arg[0] === '-') {
|
|
$operation = null;
|
|
continue;
|
|
}
|
|
|
|
if ($operation === 'move_into' || $operation === 'rename_to') {
|
|
if (!$last_arg) {
|
|
die('Expecting a previous argument' . PHP_EOL);
|
|
}
|
|
|
|
if ($operation === 'move_into') {
|
|
$last_arg_parts = preg_split('/, ?/', $last_arg);
|
|
if ($last_arg_parts === false) {
|
|
throw new AssertionError(preg_last_error_msg());
|
|
}
|
|
|
|
foreach ($last_arg_parts as $last_arg_part) {
|
|
if (strpos($last_arg_part, '::')) {
|
|
[, $identifier_name] = explode('::', $last_arg_part);
|
|
$to_refactor[$last_arg_part] = $arg . '::' . $identifier_name;
|
|
} else {
|
|
$namespace_parts = explode('\\', $last_arg_part);
|
|
$class_name = end($namespace_parts);
|
|
$to_refactor[$last_arg_part] = $arg . '\\' . $class_name;
|
|
}
|
|
}
|
|
} else {
|
|
$to_refactor[$last_arg] = $arg;
|
|
}
|
|
|
|
$last_arg = null;
|
|
$operation = null;
|
|
continue;
|
|
}
|
|
|
|
if ($operation === 'move' || $operation === 'rename') {
|
|
$last_arg = $arg;
|
|
|
|
continue;
|
|
}
|
|
|
|
die('Unexpected argument "' . $arg . '"' . PHP_EOL);
|
|
}
|
|
|
|
if (!$to_refactor) {
|
|
die('No --move or --rename arguments supplied' . PHP_EOL);
|
|
}
|
|
|
|
$config = CliUtils::initializeConfig(
|
|
$path_to_config,
|
|
$current_dir,
|
|
Report::TYPE_CONSOLE,
|
|
$first_autoloader
|
|
);
|
|
$config->setIncludeCollector($include_collector);
|
|
|
|
if ($config->resolve_from_config_file) {
|
|
$current_dir = $config->base_dir;
|
|
chdir($current_dir);
|
|
}
|
|
|
|
$threads = isset($options['threads'])
|
|
? (int)$options['threads']
|
|
: max(1, ProjectAnalyzer::getCpuCount() - 2);
|
|
|
|
$providers = new Providers(
|
|
new FileProvider(),
|
|
new ParserCacheProvider($config, false),
|
|
new FileStorageCacheProvider($config),
|
|
new ClassLikeStorageCacheProvider($config),
|
|
null,
|
|
new ProjectCacheProvider(Composer::getLockFilePath($current_dir))
|
|
);
|
|
|
|
$debug = array_key_exists('debug', $options) || array_key_exists('debug-by-line', $options);
|
|
$progress = $debug
|
|
? new DebugProgress()
|
|
: new DefaultProgress();
|
|
|
|
if (array_key_exists('debug-emitted-issues', $options)) {
|
|
$config->debug_emitted_issues = true;
|
|
}
|
|
|
|
$project_analyzer = new ProjectAnalyzer(
|
|
$config,
|
|
$providers,
|
|
new ReportOptions(),
|
|
[],
|
|
$threads,
|
|
$progress
|
|
);
|
|
|
|
if (array_key_exists('debug-by-line', $options)) {
|
|
$project_analyzer->debug_lines = true;
|
|
}
|
|
|
|
$project_analyzer->refactorCodeAfterCompletion($to_refactor);
|
|
|
|
$start_time = microtime(true);
|
|
|
|
$project_analyzer->check($current_dir);
|
|
|
|
IssueBuffer::finish($project_analyzer, false, $start_time);
|
|
}
|
|
}
|