Shell.php 3.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Symfony\Component\Console;
  11. use Symfony\Component\Console\Application;
  12. use Symfony\Component\Console\Input\StringInput;
  13. use Symfony\Component\Console\Output\ConsoleOutput;
  14. /**
  15. * A Shell wraps an Application to add shell capabilities to it.
  16. *
  17. * This class only works with a PHP compiled with readline support
  18. * (either --with-readline or --with-libedit)
  19. *
  20. * @author Fabien Potencier <fabien@symfony.com>
  21. */
  22. class Shell
  23. {
  24. private $application;
  25. private $history;
  26. private $output;
  27. /**
  28. * Constructor.
  29. *
  30. * If there is no readline support for the current PHP executable
  31. * a \RuntimeException exception is thrown.
  32. *
  33. * @param Application $application An application instance
  34. *
  35. * @throws \RuntimeException When Readline extension is not enabled
  36. */
  37. public function __construct(Application $application)
  38. {
  39. if (!function_exists('readline')) {
  40. throw new \RuntimeException('Unable to start the shell as the Readline extension is not enabled.');
  41. }
  42. $this->application = $application;
  43. $this->history = getenv('HOME').'/.history_'.$application->getName();
  44. $this->output = new ConsoleOutput();
  45. }
  46. /**
  47. * Runs the shell.
  48. */
  49. public function run()
  50. {
  51. $this->application->setAutoExit(false);
  52. $this->application->setCatchExceptions(true);
  53. readline_read_history($this->history);
  54. readline_completion_function(array($this, 'autocompleter'));
  55. $this->output->writeln($this->getHeader());
  56. while (true) {
  57. $command = readline($this->application->getName().' > ');
  58. if (false === $command) {
  59. $this->output->writeln("\n");
  60. break;
  61. }
  62. readline_add_history($command);
  63. readline_write_history($this->history);
  64. if (0 !== $ret = $this->application->run(new StringInput($command), $this->output)) {
  65. $this->output->writeln(sprintf('<error>The command terminated with an error status (%s)</error>', $ret));
  66. }
  67. }
  68. }
  69. /**
  70. * Tries to return autocompletion for the current entered text.
  71. *
  72. * @param string $text The last segment of the entered text
  73. * @param integer $position The current position
  74. */
  75. private function autocompleter($text, $position)
  76. {
  77. $info = readline_info();
  78. $text = substr($info['line_buffer'], 0, $info['end']);
  79. if ($info['point'] !== $info['end']) {
  80. return true;
  81. }
  82. // task name?
  83. if (false === strpos($text, ' ') || !$text) {
  84. return array_keys($this->application->all());
  85. }
  86. // options and arguments?
  87. try {
  88. $command = $this->application->find(substr($text, 0, strpos($text, ' ')));
  89. } catch (\Exception $e) {
  90. return true;
  91. }
  92. $list = array('--help');
  93. foreach ($command->getDefinition()->getOptions() as $option) {
  94. $list[] = '--'.$option->getName();
  95. }
  96. return $list;
  97. }
  98. /**
  99. * Returns the shell header.
  100. *
  101. * @return string The header string
  102. */
  103. protected function getHeader()
  104. {
  105. return <<<EOF
  106. Welcome to the <info>{$this->application->getName()}</info> shell (<comment>{$this->application->getVersion()}</comment>).
  107. At the prompt, type <comment>help</comment> for some help,
  108. or <comment>list</comment> to get a list of available commands.
  109. To exit the shell, type <comment>^D</comment>.
  110. EOF;
  111. }
  112. }