$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 <<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); } }