测试laravel commands的方法

引言

最近使用到laravel的consolo命令行工具,在编写命令,想写一些测试的时候,发现官方文档中并没有提到command的测试方法。花了点时间,翻墙找了资料,实践成功并记录一下,方便更多人。

测试方法

大家都知道Laravel中使用了很多Symfony的成熟组件,Laravel的console组件使用的就是Symfony/console

幸运的是,Symfony/console 组件中提供了用于command测试的CommandTester, 使用方法如下

...
use FooCommand;
use Symfony\Component\Console\Application;
use Symfony\Component\Console\Tester\CommandTester;
...

public function testSample(){
    //创建一个console测试应用平台,用来搭载测试的命令
    $application = new Application();
    
    //创建待测试的command
    $testedCommand = $this->app->make(FooCommand::class);
    //设置命令执行需要的laravel依赖
    $testedCommand->setLaravel(app());

    //添加待测试的command到测试应用上
    //同时command 也绑定 application
    $application->add($testedCommand);

    //实例化命令测试类
    $commandTester = new CommandTester($testedCommand);
    //命令输入流,对应每次交互需要提供的输入内容
    $commandTester->setInputs([
        //...
        ]);
    //执行命令
    $commandTester->execute(['command' => $testedCommand->getName()]);

    //对命令执行结果进行断言测试,主要是依靠正则判断
    //$commandTester->getDisplay() 方法可以获取命令执行后的输出结果
    $this->assertRegExp("/some reg/", $commandTester->getDisplay());

}

示例

我们现在有一个手动创建新用户的命令createUser,作用就是手动创建一个用户。

需要交互式让用户输入name,email,password,comfirm password,这些数据。

待测试的command

<?php

namespace App\Console\Commands;

use App\User;
use Illuminate\Auth\Events\Registered;
use Illuminate\Console\Command;
use Illuminate\Support\Facades\Validator;

class CreateUser extends Command
{
    /**
     * The name and signature of the console command.
     *
     * @var string
     */
    protected $signature = 'createUser';

    /**
     * The console command description.
     *
     * @var string
     */
    protected $description = 'create new user for system manually';

    /**
     * Create a new command instance.
     *
     * @return void
     */
    public function __construct()
    {
        parent::__construct();
    }

    /**
     * Execute the console command.
     *
     * @return mixed
     */
    public function handle()
    {
        $this->line($this->description);

        // 获取输入的数据
        $data = [
            'name' => $this->ask('What\'s your name?'),
            'email' => $this->ask('What\'s your email?'),
            'password' => $this->secret('What\'s your password?'),
            'password_confirmation' => $this->secret('Pleas confirm your password.')
        ];

        // 验证输入内容
        $validator = $this->makeValidator($data);
        if ($validator->fails()) {
            foreach ($validator->errors()->toArray() as $error) {
                foreach ($error as $message) {
                    $this->error($message);
                }
            }
            return;
        }

        // 向用户确认输入信息
        if (!$this->confirm('Confirm your info: ' . PHP_EOL . 'name:' . $data['name'] . PHP_EOL . 'email:' . $data['email'] . PHP_EOL . 'is this correct?')) {
            return;
        }

        // 注册
        $user = $this->create($data);
        event(new Registered($user));
        $this->line('User ' . $user->name . ' successfully registered');
    }

    /**
     * Get a validator for an incoming registration request.
     *
     * @param  array $data
     * @return \Illuminate\Contracts\Validation\Validator
     */
    protected function makeValidator($data)
    {
        return Validator::make($data, [
            'name' => 'required|string|max:255|unique:users',
            'email' => 'required|string|email|max:255|unique:users',
            'password' => 'required|string|min:6|confirmed'
        ]);
    }

    /**
     * Create a new user instance after a valid registration.
     *
     * @param  array $data
     * @return \App\User
     */
    protected function create($data)
    {
        return User::create([
            'name' => $data['name'],
            'email' => $data['email'],
            'password' => bcrypt($data['password'])
        ]);
    }
}

正确的结果

如果正确输入信息的话,会得到如下输出

$ path-to-your-app/app# php artisan createUser
create new user for system manually

 What's your name?:
 > vestin

 What's your email?:
 > [email protected]

 What's your password?:
 > 

 Pleas confirm your password.:
 > 

 Confirm your info: 
name:vestin
email:[email protected]
is this correct? (yes/no) [no]:
 > yes

User vestin successfully registered

想要测试的内容

我想要测试两块内容:

  • 数据输入验证测试

    • email有效性测试
    • password两次输入是否相同的测试
  • 正确创建用户测试

编写单元测试

<?php

namespace Tests\Unit\command;

use App\Console\Commands\CreateUser;
use Symfony\Component\Console\Application;
use Symfony\Component\Console\Tester\CommandTester;
use Tests\TestCase;
use Illuminate\Foundation\Testing\RefreshDatabase;

class CreateUserTest extends TestCase
{

    use RefreshDatabase;

    /**
     * 测试数据验证
     *
     * @return void
     */
    public function testValidation()
    {
        $application = new Application();

        $testedCommand = $this->app->make(CreateUser::class);
        $testedCommand->setLaravel(app());
        $application->add($testedCommand);

        $commandTester = new CommandTester($testedCommand);
        $commandTester->setInputs(['Vestin', 'badEmail@abc', '123456', '654321']);
        $commandTester->execute(['command' => $testedCommand->getName()]);

        // assert
        $this->assertRegExp("/The email must be a valid email address/", $commandTester->getDisplay());

        $commandTester->setInputs(['vestin', '[email protected]', '123456', '654321']);
        $commandTester->execute(['command' => $testedCommand->getName()]);

        // assert
        $this->assertRegExp("/The password confirmation does not match/", $commandTester->getDisplay());
    }

    /**
     * 测试成功注册用户
     *
     * @return void
     */
    public function testSuccess()
    {
        $application = new Application();

        $testedCommand = $this->app->make(CreateUser::class);
        $testedCommand->setLaravel(app());
        $application->add($testedCommand);

        $commandTester = new CommandTester($testedCommand);
        $commandTester->setInputs(['Vestin', '[email protected]', '123456', '123456', 'y']);
        $commandTester->execute(['command' => $testedCommand->getName()]);

        // assert
        $this->assertRegExp("/User Vestin successfully registered/", $commandTester->getDisplay());
        $this->assertDatabaseHas('users', [
            'email' => '[email protected]',
            'name' => 'Vestin'
        ]);
    }
}

执行测试

$ path-to-your-app/app#  ./vendor/bin/phpunit 
PHPUnit 6.4.3 by Sebastian Bergmann and contributors.

..                                                                  3 / 3 (100%)

Time: 659 ms, Memory: 14.00MB

参考文档

相关推荐