HelpController.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\console\controllers;
  8. use Yii;
  9. use yii\base\Application;
  10. use yii\console\Controller;
  11. use yii\console\Exception;
  12. use yii\helpers\Console;
  13. use yii\helpers\Inflector;
  14. /**
  15. * Provides help information about console commands.
  16. *
  17. * This command displays the available command list in
  18. * the application or the detailed instructions about using
  19. * a specific command.
  20. *
  21. * This command can be used as follows on command line:
  22. *
  23. * ```
  24. * yii help [command name]
  25. * ```
  26. *
  27. * In the above, if the command name is not provided, all
  28. * available commands will be displayed.
  29. *
  30. * @property-read array $commands All available command names.
  31. *
  32. * @author Qiang Xue <qiang.xue@gmail.com>
  33. * @since 2.0
  34. */
  35. class HelpController extends Controller
  36. {
  37. /**
  38. * Displays available commands or the detailed information
  39. * about a particular command.
  40. *
  41. * @param string|null $command The name of the command to show help about.
  42. * If not provided, all available commands will be displayed.
  43. * @return int the exit status
  44. * @throws Exception if the command for help is unknown
  45. */
  46. public function actionIndex($command = null)
  47. {
  48. if ($command !== null) {
  49. $result = Yii::$app->createController($command);
  50. if ($result === false) {
  51. $name = $this->ansiFormat($command, Console::FG_YELLOW);
  52. throw new Exception("No help for unknown command \"$name\".");
  53. }
  54. list($controller, $actionID) = $result;
  55. $actions = $this->getActions($controller);
  56. if ($actionID !== '' || count($actions) === 1 && $actions[0] === $controller->defaultAction) {
  57. $this->getSubCommandHelp($controller, $actionID);
  58. } else {
  59. $this->getCommandHelp($controller);
  60. }
  61. } else {
  62. $this->getDefaultHelp();
  63. }
  64. }
  65. /**
  66. * List all available controllers and actions in machine readable format.
  67. * This is used for shell completion.
  68. * @since 2.0.11
  69. */
  70. public function actionList()
  71. {
  72. foreach ($this->getCommandDescriptions() as $command => $description) {
  73. $result = Yii::$app->createController($command);
  74. /** @var $controller Controller */
  75. list($controller, $actionID) = $result;
  76. $actions = $this->getActions($controller);
  77. $prefix = $controller->getUniqueId();
  78. if ($controller->createAction($controller->defaultAction) !== null) {
  79. $this->stdout("$prefix\n");
  80. }
  81. foreach ($actions as $action) {
  82. $this->stdout("$prefix/$action\n");
  83. }
  84. }
  85. }
  86. /**
  87. * List all available options for the $action in machine readable format.
  88. * This is used for shell completion.
  89. *
  90. * @param string $action route to action
  91. * @since 2.0.11
  92. */
  93. public function actionListActionOptions($action)
  94. {
  95. $result = Yii::$app->createController($action);
  96. if ($result === false || !($result[0] instanceof Controller)) {
  97. return;
  98. }
  99. /** @var Controller $controller */
  100. list($controller, $actionID) = $result;
  101. $action = $controller->createAction($actionID);
  102. if ($action === null) {
  103. return;
  104. }
  105. foreach ($controller->getActionArgsHelp($action) as $argument => $help) {
  106. $description = preg_replace("~\R~", '', addcslashes($help['comment'], ':')) ?: $argument;
  107. $this->stdout($argument . ':' . $description . "\n");
  108. }
  109. $this->stdout("\n");
  110. foreach ($controller->getActionOptionsHelp($action) as $argument => $help) {
  111. $description = preg_replace("~\R~", '', addcslashes($help['comment'], ':'));
  112. $this->stdout('--' . $argument . ($description ? ':' . $description : '') . "\n");
  113. }
  114. }
  115. /**
  116. * Displays usage information for $action.
  117. *
  118. * @param string $action route to action
  119. * @since 2.0.11
  120. */
  121. public function actionUsage($action)
  122. {
  123. $result = Yii::$app->createController($action);
  124. if ($result === false || !($result[0] instanceof Controller)) {
  125. return;
  126. }
  127. /** @var Controller $controller */
  128. list($controller, $actionID) = $result;
  129. $action = $controller->createAction($actionID);
  130. if ($action === null) {
  131. return;
  132. }
  133. $scriptName = $this->getScriptName();
  134. if ($action->id === $controller->defaultAction) {
  135. $this->stdout($scriptName . ' ' . $this->ansiFormat($controller->getUniqueId(), Console::FG_YELLOW));
  136. } else {
  137. $this->stdout($scriptName . ' ' . $this->ansiFormat($action->getUniqueId(), Console::FG_YELLOW));
  138. }
  139. foreach ($controller->getActionArgsHelp($action) as $name => $arg) {
  140. if ($arg['required']) {
  141. $this->stdout(' <' . $name . '>', Console::FG_CYAN);
  142. } else {
  143. $this->stdout(' [' . $name . ']', Console::FG_CYAN);
  144. }
  145. }
  146. $this->stdout("\n");
  147. }
  148. /**
  149. * Returns all available command names.
  150. * @return array all available command names
  151. */
  152. public function getCommands()
  153. {
  154. $commands = $this->getModuleCommands(Yii::$app);
  155. sort($commands);
  156. return array_filter(array_unique($commands), function ($command) {
  157. $result = Yii::$app->createController($command);
  158. if ($result === false || !$result[0] instanceof Controller) {
  159. return false;
  160. }
  161. list($controller, $actionID) = $result;
  162. $actions = $this->getActions($controller);
  163. return $actions !== [];
  164. });
  165. }
  166. /**
  167. * Returns an array of commands an their descriptions.
  168. * @return array all available commands as keys and their description as values.
  169. */
  170. protected function getCommandDescriptions()
  171. {
  172. $descriptions = [];
  173. foreach ($this->getCommands() as $command) {
  174. $result = Yii::$app->createController($command);
  175. /** @var Controller $controller */
  176. list($controller, $actionID) = $result;
  177. $descriptions[$command] = $controller->getHelpSummary();
  178. }
  179. return $descriptions;
  180. }
  181. /**
  182. * Returns all available actions of the specified controller.
  183. * @param Controller $controller the controller instance
  184. * @return array all available action IDs.
  185. */
  186. public function getActions($controller)
  187. {
  188. $actions = array_keys($controller->actions());
  189. $class = new \ReflectionClass($controller);
  190. foreach ($class->getMethods() as $method) {
  191. $name = $method->getName();
  192. if ($name !== 'actions' && $method->isPublic() && !$method->isStatic() && strncmp($name, 'action', 6) === 0) {
  193. $actions[] = $this->camel2id(substr($name, 6));
  194. }
  195. }
  196. sort($actions);
  197. return array_unique($actions);
  198. }
  199. /**
  200. * Returns available commands of a specified module.
  201. * @param \yii\base\Module $module the module instance
  202. * @return array the available command names
  203. */
  204. protected function getModuleCommands($module)
  205. {
  206. $prefix = $module instanceof Application ? '' : $module->getUniqueId() . '/';
  207. $commands = [];
  208. foreach (array_keys($module->controllerMap) as $id) {
  209. $commands[] = $prefix . $id;
  210. }
  211. foreach ($module->getModules() as $id => $child) {
  212. if (($child = $module->getModule($id)) === null) {
  213. continue;
  214. }
  215. foreach ($this->getModuleCommands($child) as $command) {
  216. $commands[] = $command;
  217. }
  218. }
  219. $controllerPath = $module->getControllerPath();
  220. if (is_dir($controllerPath)) {
  221. $iterator = new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($controllerPath, \RecursiveDirectoryIterator::KEY_AS_PATHNAME));
  222. $iterator = new \RegexIterator($iterator, '/.*Controller\.php$/', \RecursiveRegexIterator::GET_MATCH);
  223. foreach ($iterator as $matches) {
  224. $file = $matches[0];
  225. $relativePath = str_replace($controllerPath, '', $file);
  226. $class = strtr($relativePath, [
  227. '/' => '\\',
  228. '.php' => '',
  229. ]);
  230. $controllerClass = $module->controllerNamespace . $class;
  231. if ($this->validateControllerClass($controllerClass)) {
  232. $dir = ltrim(pathinfo($relativePath, PATHINFO_DIRNAME), '\\/');
  233. $command = Inflector::camel2id(substr(basename($file), 0, -14), '-', true);
  234. if (!empty($dir)) {
  235. $command = $dir . '/' . $command;
  236. }
  237. $commands[] = $prefix . $command;
  238. }
  239. }
  240. }
  241. return $commands;
  242. }
  243. /**
  244. * Validates if the given class is a valid console controller class.
  245. * @param string $controllerClass
  246. * @return bool
  247. */
  248. protected function validateControllerClass($controllerClass)
  249. {
  250. if (class_exists($controllerClass)) {
  251. $class = new \ReflectionClass($controllerClass);
  252. return !$class->isAbstract() && $class->isSubclassOf('yii\console\Controller');
  253. }
  254. return false;
  255. }
  256. /**
  257. * Displays all available commands.
  258. */
  259. protected function getDefaultHelp()
  260. {
  261. $commands = $this->getCommandDescriptions();
  262. $this->stdout($this->getDefaultHelpHeader());
  263. if (empty($commands)) {
  264. $this->stdout("\nNo commands are found.\n\n", Console::BOLD);
  265. return;
  266. }
  267. $this->stdout("\nThe following commands are available:\n\n", Console::BOLD);
  268. $maxLength = 0;
  269. foreach ($commands as $command => $description) {
  270. $result = Yii::$app->createController($command);
  271. /** @var $controller Controller */
  272. list($controller, $actionID) = $result;
  273. $actions = $this->getActions($controller);
  274. $prefix = $controller->getUniqueId();
  275. foreach ($actions as $action) {
  276. $string = $prefix . '/' . $action;
  277. if ($action === $controller->defaultAction) {
  278. $string .= ' (default)';
  279. }
  280. $maxLength = max($maxLength, strlen($string));
  281. }
  282. }
  283. foreach ($commands as $command => $description) {
  284. $result = Yii::$app->createController($command);
  285. list($controller, $actionID) = $result;
  286. $actions = $this->getActions($controller);
  287. $this->stdout('- ' . $this->ansiFormat($command, Console::FG_YELLOW));
  288. $this->stdout(str_repeat(' ', $maxLength + 4 - strlen($command)));
  289. $this->stdout(Console::wrapText($description, $maxLength + 4 + 2), Console::BOLD);
  290. $this->stdout("\n");
  291. $prefix = $controller->getUniqueId();
  292. foreach ($actions as $action) {
  293. $string = ' ' . $prefix . '/' . $action;
  294. $this->stdout(' ' . $this->ansiFormat($string, Console::FG_GREEN));
  295. if ($action === $controller->defaultAction) {
  296. $string .= ' (default)';
  297. $this->stdout(' (default)', Console::FG_YELLOW);
  298. }
  299. $summary = $controller->getActionHelpSummary($controller->createAction($action));
  300. if ($summary !== '') {
  301. $this->stdout(str_repeat(' ', $maxLength + 4 - strlen($string)));
  302. $this->stdout(Console::wrapText($summary, $maxLength + 4 + 2));
  303. }
  304. $this->stdout("\n");
  305. }
  306. $this->stdout("\n");
  307. }
  308. $scriptName = $this->getScriptName();
  309. $this->stdout("\nTo see the help of each command, enter:\n", Console::BOLD);
  310. $this->stdout("\n $scriptName " . $this->ansiFormat('help', Console::FG_YELLOW) . ' '
  311. . $this->ansiFormat('<command-name>', Console::FG_CYAN) . "\n\n");
  312. }
  313. /**
  314. * Displays the overall information of the command.
  315. * @param Controller $controller the controller instance
  316. */
  317. protected function getCommandHelp($controller)
  318. {
  319. $controller->color = $this->color;
  320. $this->stdout("\nDESCRIPTION\n", Console::BOLD);
  321. $comment = $controller->getHelp();
  322. if ($comment !== '') {
  323. $this->stdout("\n$comment\n\n");
  324. }
  325. $actions = $this->getActions($controller);
  326. if (!empty($actions)) {
  327. $this->stdout("\nSUB-COMMANDS\n\n", Console::BOLD);
  328. $prefix = $controller->getUniqueId();
  329. $maxlen = 5;
  330. foreach ($actions as $action) {
  331. $len = strlen($prefix . '/' . $action) + 2 + ($action === $controller->defaultAction ? 10 : 0);
  332. $maxlen = max($maxlen, $len);
  333. }
  334. foreach ($actions as $action) {
  335. $this->stdout('- ' . $this->ansiFormat($prefix . '/' . $action, Console::FG_YELLOW));
  336. $len = strlen($prefix . '/' . $action) + 2;
  337. if ($action === $controller->defaultAction) {
  338. $this->stdout(' (default)', Console::FG_GREEN);
  339. $len += 10;
  340. }
  341. $summary = $controller->getActionHelpSummary($controller->createAction($action));
  342. if ($summary !== '') {
  343. $this->stdout(str_repeat(' ', $maxlen - $len + 2) . Console::wrapText($summary, $maxlen + 2));
  344. }
  345. $this->stdout("\n");
  346. }
  347. $scriptName = $this->getScriptName();
  348. $this->stdout("\nTo see the detailed information about individual sub-commands, enter:\n");
  349. $this->stdout("\n $scriptName " . $this->ansiFormat('help', Console::FG_YELLOW) . ' '
  350. . $this->ansiFormat('<sub-command>', Console::FG_CYAN) . "\n\n");
  351. }
  352. }
  353. /**
  354. * Displays the detailed information of a command action.
  355. * @param Controller $controller the controller instance
  356. * @param string $actionID action ID
  357. * @throws Exception if the action does not exist
  358. */
  359. protected function getSubCommandHelp($controller, $actionID)
  360. {
  361. $action = $controller->createAction($actionID);
  362. if ($action === null) {
  363. $name = $this->ansiFormat(rtrim($controller->getUniqueId() . '/' . $actionID, '/'), Console::FG_YELLOW);
  364. throw new Exception("No help for unknown sub-command \"$name\".");
  365. }
  366. $description = $controller->getActionHelp($action);
  367. if ($description !== '') {
  368. $this->stdout("\nDESCRIPTION\n", Console::BOLD);
  369. $this->stdout("\n$description\n\n");
  370. }
  371. $this->stdout("\nUSAGE\n\n", Console::BOLD);
  372. $scriptName = $this->getScriptName();
  373. if ($action->id === $controller->defaultAction) {
  374. $this->stdout($scriptName . ' ' . $this->ansiFormat($controller->getUniqueId(), Console::FG_YELLOW));
  375. } else {
  376. $this->stdout($scriptName . ' ' . $this->ansiFormat($action->getUniqueId(), Console::FG_YELLOW));
  377. }
  378. $args = $controller->getActionArgsHelp($action);
  379. foreach ($args as $name => $arg) {
  380. if ($arg['required']) {
  381. $this->stdout(' <' . $name . '>', Console::FG_CYAN);
  382. } else {
  383. $this->stdout(' [' . $name . ']', Console::FG_CYAN);
  384. }
  385. }
  386. $options = $controller->getActionOptionsHelp($action);
  387. $options[\yii\console\Application::OPTION_APPCONFIG] = [
  388. 'type' => 'string',
  389. 'default' => null,
  390. 'comment' => "custom application configuration file path.\nIf not set, default application configuration is used.",
  391. ];
  392. ksort($options);
  393. if (!empty($options)) {
  394. $this->stdout(' [...options...]', Console::FG_RED);
  395. }
  396. $this->stdout("\n\n");
  397. if (!empty($args)) {
  398. foreach ($args as $name => $arg) {
  399. $this->stdout($this->formatOptionHelp(
  400. '- ' . $this->ansiFormat($name, Console::FG_CYAN),
  401. $arg['required'],
  402. $arg['type'],
  403. $arg['default'],
  404. $arg['comment']) . "\n\n");
  405. }
  406. }
  407. if (!empty($options)) {
  408. $this->stdout("\nOPTIONS\n\n", Console::BOLD);
  409. foreach ($options as $name => $option) {
  410. $this->stdout($this->formatOptionHelp(
  411. $this->ansiFormat('--' . $name . $this->formatOptionAliases($controller, $name),
  412. Console::FG_RED, empty($option['required']) ? Console::FG_RED : Console::BOLD),
  413. !empty($option['required']),
  414. $option['type'],
  415. $option['default'],
  416. $option['comment']) . "\n\n");
  417. }
  418. }
  419. }
  420. /**
  421. * Generates a well-formed string for an argument or option.
  422. * @param string $name the name of the argument or option
  423. * @param bool $required whether the argument is required
  424. * @param string $type the type of the option or argument
  425. * @param mixed $defaultValue the default value of the option or argument
  426. * @param string $comment comment about the option or argument
  427. * @return string the formatted string for the argument or option
  428. */
  429. protected function formatOptionHelp($name, $required, $type, $defaultValue, $comment)
  430. {
  431. $comment = trim((string)$comment);
  432. $type = trim((string)$type);
  433. if (strncmp($type, 'bool', 4) === 0) {
  434. $type = 'boolean, 0 or 1';
  435. }
  436. if ($defaultValue !== null && !is_array($defaultValue)) {
  437. if ($type === null) {
  438. $type = gettype($defaultValue);
  439. }
  440. if (is_bool($defaultValue)) {
  441. // show as integer to avoid confusion
  442. $defaultValue = (int) $defaultValue;
  443. }
  444. if (is_string($defaultValue)) {
  445. $defaultValue = "'" . $defaultValue . "'";
  446. } else {
  447. $defaultValue = var_export($defaultValue, true);
  448. }
  449. $doc = "$type (defaults to $defaultValue)";
  450. } else {
  451. $doc = $type;
  452. }
  453. if ($doc === '') {
  454. $doc = $comment;
  455. } elseif ($comment !== '') {
  456. $doc .= "\n" . preg_replace('/^/m', ' ', $comment);
  457. }
  458. $name = $required ? "$name (required)" : $name;
  459. return $doc === '' ? $name : "$name: $doc";
  460. }
  461. /**
  462. * @param Controller $controller the controller instance
  463. * @param string $option the option name
  464. * @return string the formatted string for the alias argument or option
  465. * @since 2.0.8
  466. */
  467. protected function formatOptionAliases($controller, $option)
  468. {
  469. foreach ($controller->optionAliases() as $name => $value) {
  470. if (Inflector::camel2id($value, '-', true) === $option) {
  471. return ', -' . $name;
  472. }
  473. }
  474. return '';
  475. }
  476. /**
  477. * @return string the name of the cli script currently running.
  478. */
  479. protected function getScriptName()
  480. {
  481. return basename(Yii::$app->request->scriptFile);
  482. }
  483. /**
  484. * Return a default help header.
  485. * @return string default help header.
  486. * @since 2.0.11
  487. */
  488. protected function getDefaultHelpHeader()
  489. {
  490. return "\nThis is Yii version " . \Yii::getVersion() . ".\n";
  491. }
  492. /**
  493. * Converts a CamelCase action name into an ID in lowercase.
  494. * Words in the ID are concatenated using the specified character '-'.
  495. * For example, 'CreateUser' will be converted to 'create-user'.
  496. * @param string $name the string to be converted
  497. * @return string the resulting ID
  498. */
  499. private function camel2id($name)
  500. {
  501. return mb_strtolower(trim(preg_replace('/\p{Lu}/u', '-\0', $name), '-'), 'UTF-8');
  502. }
  503. }