命令行

Hyperf 的命令行默认由 hyperf/command 组件提供,而该组件本身也是基于 symfony/console 的抽象。

安装

通常来说该组件会默认存在,但如果您希望用于非 Hyperf 项目,也可通过下面的命令依赖 hyperf/command 组件:

  1. composer require hyperf/command

查看命令列表

直接运行 php bin/hyperf.php 不带任何的参数即为输出命令列表。

自定义命令

生成命令

如果你有安装 hyperf/devtool 组件的话,可以通过 gen:command 命令来生成一个自定义命令:

  1. php bin/hyperf.php gen:command FooCommand

执行上述命令后,便会在 app/Command 文件夹内生成一个配置好的 FooCommand 类了。

定义命令

定义该命令类所对应的命令有两种形式,一种是通过 $name 属性定义,另一种是通过构造函数传参来定义,我们通过代码示例来演示一下,假设我们希望定义该命令类的命令为 foo:hello

$name 属性定义:

  1. <?php
  2. declare(strict_types=1);
  3. namespace App\Command;
  4. use Hyperf\Command\Command as HyperfCommand;
  5. use Hyperf\Command\Annotation\Command;
  6. /**
  7. * @Command
  8. */
  9. class FooCommand extends HyperfCommand
  10. {
  11. /**
  12. * 执行的命令行
  13. *
  14. * @var string
  15. */
  16. protected $name = 'foo:hello';
  17. }

构造函数传参定义:

  1. <?php
  2. declare(strict_types=1);
  3. namespace App\Command;
  4. use Hyperf\Command\Command as HyperfCommand;
  5. use Hyperf\Command\Annotation\Command;
  6. /**
  7. * @Command
  8. */
  9. class FooCommand extends HyperfCommand
  10. {
  11. public function __construct()
  12. {
  13. parent::__construct('foo:hello');
  14. }
  15. }

定义命令类逻辑

命令类实际运行的逻辑是取决于 handle 方法内的代码,也就意味着 handle 方法就是命令的入口。

  1. <?php
  2. declare(strict_types=1);
  3. namespace App\Command;
  4. use Hyperf\Command\Command as HyperfCommand;
  5. use Hyperf\Command\Annotation\Command;
  6. /**
  7. * @Command
  8. */
  9. class FooCommand extends HyperfCommand
  10. {
  11. /**
  12. * 执行的命令行
  13. *
  14. * @var string
  15. */
  16. protected $name = 'foo:hello';
  17. public function handle()
  18. {
  19. // 通过内置方法 line 在 Console 输出 Hello Hyperf.
  20. $this->line('Hello Hyperf.', 'info');
  21. }
  22. }

定义命令类的参数

在编写命令时,通常是通过 参数选项 来收集用户的输入的,在收集一个用户输入前,必须对该 参数选项 进行定义。

参数

假设我们希望定义一个 name 参数,然后通过传递任意字符串如 Hyperf 于命令一起并执行 php bin/hyperf.php foo:hello Hyperf 输出 Hello Hyperf,我们通过代码来演示一下:

  1. <?php
  2. declare(strict_types=1);
  3. namespace App\Command;
  4. use Hyperf\Command\Annotation\Command;
  5. use Hyperf\Command\Command as HyperfCommand;
  6. use Symfony\Component\Console\Input\InputArgument;
  7. /**
  8. * @Command
  9. */
  10. class FooCommand extends HyperfCommand
  11. {
  12. /**
  13. * 执行的命令行
  14. *
  15. * @var string
  16. */
  17. protected $name = 'foo:hello';
  18. public function handle()
  19. {
  20. // 从 $input 获取 name 参数
  21. $argument = $this->input->getArgument('name') ?? 'World';
  22. $this->line('Hello ' . $argument, 'info');
  23. }
  24. protected function getArguments()
  25. {
  26. return [
  27. ['name', InputArgument::OPTIONAL, '这里是对这个参数的解释']
  28. ];
  29. }
  30. }

执行 php bin/hyperf.php foo:hello Hyperf 我们就能看到输出了 Hello Hyperf 了。

命令常用配置介绍

以下代码皆只修改 configurehandle 中的内容。

设置 Help

  1. public function configure()
  2. {
  3. parent::configure();
  4. $this->setHelp('Hyperf 自定义命令演示');
  5. }
  6. $ php bin/hyperf.php demo:command --help
  7. ...
  8. Help:
  9. Hyperf 自定义命令演示

设置 Description

  1. public function configure()
  2. {
  3. parent::configure();
  4. $this->setDescription('Hyperf Demo Command');
  5. }
  6. $ php bin/hyperf.php demo:command --help
  7. ...
  8. Description:
  9. Hyperf Demo Command

设置 Usage

  1. public function configure()
  2. {
  3. parent::configure();
  4. $this->addUsage('--name 演示代码');
  5. }
  6. $ php bin/hyperf.php demo:command --help
  7. ...
  8. Usage:
  9. demo:command
  10. demo:command --name 演示代码

设置参数

参数支持以下模式。

模式 备注
InputArgument::REQUIRED 1 参数必填,此种模式 default 字段无效
InputArgument::OPTIONAL 2 参数可选,常配合 default 使用
InputArgument::IS_ARRAY 4 数组类型

可选类型

  1. public function configure()
  2. {
  3. parent::configure();
  4. $this->addArgument('name', InputArgument::OPTIONAL, '姓名', 'Hyperf');
  5. }
  6. public function handle()
  7. {
  8. $this->line($this->input->getArgument('name'));
  9. }
  10. $ php bin/hyperf.php demo:command
  11. ...
  12. Hyperf
  13. $ php bin/hyperf.php demo:command Swoole
  14. ...
  15. Swoole

数组类型

  1. public function configure()
  2. {
  3. parent::configure();
  4. $this->addArgument('name', InputArgument::IS_ARRAY, '姓名');
  5. }
  6. public function handle()
  7. {
  8. var_dump($this->input->getArgument('name'));
  9. }
  10. $ php bin/hyperf.php demo:command Hyperf Swoole
  11. ...
  12. array(2) {
  13. [0]=>
  14. string(6) "Hyperf"
  15. [1]=>
  16. string(6) "Swoole"
  17. }

设置选项

选项支持以下模式。

模式 备注
InputOption::VALUE_NONE 1 是否传入可选项 default 字段无效
InputOption::VALUE_REQUIRED 2 选项必填
InputOption::VALUE_OPTIONAL 4 选项可选
InputOption::VALUE_IS_ARRAY 8 选项数组

是否传入可选项

  1. public function configure()
  2. {
  3. parent::configure();
  4. $this->addOption('opt', 'o', InputOption::VALUE_NONE, '是否优化');
  5. }
  6. public function handle()
  7. {
  8. var_dump($this->input->getOption('opt'));
  9. }
  10. $ php bin/hyperf.php demo:command
  11. bool(false)
  12. $ php bin/hyperf.php demo:command -o
  13. bool(true)
  14. $ php bin/hyperf.php demo:command --opt
  15. bool(true)

选项必填和可选

VALUE_OPTIONAL 在单独使用上与 VALUE_REQUIRED 并无二致

  1. public function configure()
  2. {
  3. parent::configure();
  4. $this->addOption('name', 'N', InputOption::VALUE_REQUIRED, '姓名', 'Hyperf');
  5. }
  6. public function handle()
  7. {
  8. var_dump($this->input->getOption('name'));
  9. }
  10. $ php bin/hyperf.php demo:command
  11. string(6) "Hyperf"
  12. $ php bin/hyperf.php demo:command --name Swoole
  13. string(6) "Swoole"

选项数组

VALUE_IS_ARRAYVALUE_OPTIONAL 配合使用,可以达到传入多个 Option 的效果。

  1. public function configure()
  2. {
  3. parent::configure();
  4. $this->addOption('name', 'N', InputOption::VALUE_IS_ARRAY | InputOption::VALUE_OPTIONAL, '姓名');
  5. }
  6. public function handle()
  7. {
  8. var_dump($this->input->getOption('name'));
  9. }
  10. $ php bin/hyperf.php demo:command
  11. array(0) {
  12. }
  13. $ php bin/hyperf.php demo:command --name Hyperf --name Swoole
  14. array(2) {
  15. [0]=>
  16. string(6) "Hyperf"
  17. [1]=>
  18. string(6) "Swoole"
  19. }