iT邦幫忙

2022 iThome 鐵人賽

DAY 18
0
DevOps

自動化測試大作戰系列 第 18

Mocking(三):Event

  • 分享至 

  • xImage
  •  

Medium 清新閱讀版連結

今天我們會接續環繞著 「Mocking」 這個主題。

在 Laravel 中,有幾個類別的 Mocking 方式,與前兩天所介紹的方式有所不同,在這次的系列文章中,會和大家介紹 Event、Mail、Queue、Storage、HTTP 這幾個類別的特殊 Mocking 方式。

今天就先來看 Event Mocking 吧!

Event Mocking 函數

  • Event::fake():當我們希望在執行測試目標行為時, 想驗證是否有觸發到 Event ,但又不要真的觸發 Event 的執行時,可在測試程式碼中呼叫此函數。
  • Event::assertDispatched():可驗證指定的 Event 是否會被觸發。需在執行 Event::fake() 後方可使用。
  • Event::assertNotDispatched():可驗證指定的 Event 是否不會被觸發。需在執行 Event::fake() 後方可使用。
  • Event::assertNothingDispatched():可驗正是否無 Event 被觸發。需在執行 Event::fake() 後方可使用。

(實際上 Event 還有其他 Mocking 函數可用,不過以上應該是最常用的幾個了。

接下來讓我們實際演練看看吧!

範例:使用者登入

  • 測試目標:登入端點

    • database/migrations/2014_10_12_000000_create_users_table.php

      <?php
      
      use Illuminate\Database\Migrations\Migration;
      use Illuminate\Database\Schema\Blueprint;
      use Illuminate\Support\Facades\Schema;
      
      return new class extends Migration
      {
          /**
           * Run the migrations.
           *
           * @return void
           */
          public function up()
          {
              Schema::create('users', function (Blueprint $table) {
                  $table->id();
                  $table->string('name');
                  $table->string('email')->unique();
                  $table->timestamp('email_verified_at')->nullable();
                  $table->string('password');
      						// 新增 api_token 欄位
                  $table->string('api_token', 32)->nullable();
                  $table->rememberToken();
                  $table->timestamps();
              });
          }
      
          /**
           * Reverse the migrations.
           *
           * @return void
           */
          public function down()
          {
              Schema::dropIfExists('users');
          }
      };
      
    • routes\web.php

      use App\Events\UserLoggedIn;
      use App\Models\User;
      use Illuminate\Http\Request;
      use Illuminate\Support\Facades\Route;
      use Illuminate\Support\Str;
      
      Route::post('/login', function (Request $request) {
          $email = $request->input('email');
          $password = $request->input('password');
      
          $user = User::where([
              'email' => $email,
              'password' => $password,
          ])->first();
      
          if (empty($user)) {
              return response()->json([], 401);
          }
      
          event(UserLoggedIn::class);
      
          $user->api_token = Str::random(32);
          $user->save();
      
          return response()->json(['token' => $user->api_token]);
      })->name('login');	
      
  • 測試程式碼:

    <?php
    
    namespace Tests\Feature;
    
    use App\Events\UserLoggedIn;
    use App\Models\User;
    use Illuminate\Foundation\Testing\RefreshDatabase;
    use Illuminate\Support\Facades\Event;
    use Illuminate\Support\Str;
    use Tests\TestCase;
    
    class EventTest extends TestCase
    {
        use RefreshDatabase;
    
        public function testLoginSuccess()
        {
            $user = User::factory()->create([
                'password' => Str::random(10),
            ]);
    
            $payload = [
                'email' => $user->email,
                'password' => $user->password,
            ];
    
            Event::fake();
    
            $response = $this->post(route('login'), $payload);
    
            $user->refresh();
    
            $response->assertJson([
                'api_token' => $user->api_token
            ]);
    
            Event::assertDispatched(UserLoggedIn::class, 1);
        }
    
        public function testLoginFailed()
        {
            $user = User::factory()->create([
                'password' => Str::random(10),
            ]);
    
            $payload = [
                'email' => $user->email,
                'password' => $user->password . 'x',
            ];
    
            Event::fake();
    
            $response = $this->post(route('login'), $payload);
    
            $response->assertUnauthorized();
    
            Event::assertNotDispatched(UserLoggedIn::class);
        }
    }
    

    以上測試程式碼,測試了 2 種測試案例:

    • testLoginSuccess():在這個測試案例函數中,我們驗證了當使用者成功登入時,是否會觸發 UserLoggedIn 這個 Event的發出。
    • testLoginFailed():在這個測試案例函數中,我們驗證了當使用者登入失敗時,是否不會觸發 UserLoggedIn 這個 Event的發出。

以上就是今天所介紹的 Event Mocking,大家可以多加演練。

明天讓我們來看看 Mail Mocking。

參考資料


上一篇
Mocking(二)
下一篇
Mocking(四):Mail
系列文
自動化測試大作戰31
圖片
  直播研討會
圖片
{{ item.channelVendor }} {{ item.webinarstarted }} |
{{ formatDate(item.duration) }}
直播中

尚未有邦友留言

立即登入留言