<?php /** * This file is part of the CodeIgniter 4 framework. * * (c) CodeIgniter Foundation <admin@codeigniter.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace CodeIgniter\CLI; use CodeIgniter\Log\Logger; use Config\Services; use ReflectionClass; use ReflectionException; /** * Class Commands * * Core functionality for running, listing, etc commands. */ class Commands { /** * The found commands. * * @var array */ protected $commands = []; /** * Logger instance. * * @var Logger */ protected $logger; /** * Constructor * * @param Logger|null $logger */ public function __construct($logger = null) { $this->logger = $logger ?? Services::logger(); } /** * Runs a command given * * @param string $command * @param array $params */ public function run(string $command, array $params) { $this->discoverCommands(); if (! $this->verifyCommand($command, $this->commands)) { return; } // The file would have already been loaded during the // createCommandList function... $className = $this->commands[$command]['class']; $class = new $className($this->logger, $this); return $class->run($params); } /** * Provide access to the list of commands. * * @return array */ public function getCommands() { $this->discoverCommands(); return $this->commands; } /** * Discovers all commands in the framework and within user code, * and collects instances of them to work with. */ public function discoverCommands() { if (! empty($this->commands)) { return; } $files = service('locator')->listFiles('Commands/'); // If no matching command files were found, bail if (empty($files)) { // This should never happen in unit testing. // if it does, we have far bigger problems! // @codeCoverageIgnoreStart return; // @codeCoverageIgnoreEnd } // Loop over each file checking to see if a command with that // alias exists in the class. If so, return it. Otherwise, try the next. foreach ($files as $file) { $className = Services::locator()->findQualifiedNameFromPath($file); if (empty($className) || ! class_exists($className)) { continue; } try { $class = new ReflectionClass($className); if (! $class->isInstantiable() || ! $class->isSubclassOf(BaseCommand::class)) { continue; } $class = new $className($this->logger, $this); // Store it! if (! is_null($class->group)) { $this->commands[$class->name] = [ 'class' => $className, 'file' => $file, 'group' => $class->group, 'description' => $class->description, ]; } $class = null; unset($class); } catch (ReflectionException $e) { $this->logger->error($e->getMessage()); } } asort($this->commands); } /** * Verifies if the command being sought is found * in the commands list. * * @param string $command * @param array $commands * * @return boolean */ public function verifyCommand(string $command, array $commands): bool { if (isset($commands[$command])) { return true; } $message = lang('CLI.commandNotFound', [$command]); if ($alternatives = $this->getCommandAlternatives($command, $commands)) { if (count($alternatives) === 1) { $message .= "\n\n" . lang('CLI.altCommandSingular') . "\n "; } else { $message .= "\n\n" . lang('CLI.altCommandPlural') . "\n "; } $message .= implode("\n ", $alternatives); } CLI::error($message); CLI::newLine(); return false; } /** * Finds alternative of `$name` among collection * of commands. * * @param string $name * @param array $collection * * @return array */ protected function getCommandAlternatives(string $name, array $collection): array { $alternatives = []; foreach ($collection as $commandName => $attributes) { $lev = levenshtein($name, $commandName); if ($lev <= strlen($commandName) / 3 || strpos($commandName, $name) !== false) { $alternatives[$commandName] = $lev; } } ksort($alternatives, SORT_NATURAL | SORT_FLAG_CASE); return array_keys($alternatives); } }