測試laravel commands的方法

Vestin發表於2019-02-16

引言

最近使用到laravel的consolo命令列工具,在編寫命令,想寫一些測試的時候,發現官方文件中並沒有提到command的測試方法。花了點時間,翻牆找了資料,實踐成功並記錄一下,方便更多人。

測試方法

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

幸運的是,Symfony/console 元件中提供了用於command測試的CommandTester, 使用方法如下

...
use FooCommand;
use SymfonyComponentConsoleApplication;
use SymfonyComponentConsoleTesterCommandTester;
...

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 AppConsoleCommands;

use AppUser;
use IlluminateAuthEventsRegistered;
use IlluminateConsoleCommand;
use IlluminateSupportFacadesValidator;

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 IlluminateContractsValidationValidator
     */
    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 AppUser
     */
    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?:
 > correct@abc.com

 What`s your password?:
 > 

 Pleas confirm your password.:
 > 

 Confirm your info: 
name:vestin
email:correct@abc.com
is this correct? (yes/no) [no]:
 > yes

User vestin successfully registered

想要測試的內容

我想要測試兩塊內容:

  • 資料輸入驗證測試

    • email有效性測試
    • password兩次輸入是否相同的測試
  • 正確建立使用者測試

編寫單元測試

<?php

namespace TestsUnitcommand;

use AppConsoleCommandsCreateUser;
use SymfonyComponentConsoleApplication;
use SymfonyComponentConsoleTesterCommandTester;
use TestsTestCase;
use IlluminateFoundationTestingRefreshDatabase;

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`, `correct@abc.com`, `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`, `correct@abc.com`, `123456`, `123456`, `y`]);
        $commandTester->execute([`command` => $testedCommand->getName()]);

        // assert
        $this->assertRegExp("/User Vestin successfully registered/", $commandTester->getDisplay());
        $this->assertDatabaseHas(`users`, [
            `email` => `correct@abc.com`,
            `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

參考文件

相關文章