iT邦幫忙

2022 iThome 鐵人賽

DAY 19
1

Medium 清新閱讀版連結

今天來看 Mail Mocking 吧!

Mail Mocking 函數

  • Mail::fake():當我們希望在執行測試目標行為時, 想驗證是否有觸發到發送 Email ,但又不要真的觸發 Email 的寄送時,可在測試程式碼中呼叫此函數。
  • Mail::assertSent():可驗證指定的 Mailable 類別是否會被觸發寄送。需在執行 Mail::fake() 後方可使用。
  • Mail::assertNotSent()。可驗證指定的 Mailable 類別是否不會被觸發寄送。需在執行 Mail::fake() 後方可使用。
  • Mail::assertNothingSent()。可驗證是否無 Mailable 類別被觸發寄送。需在執行 Mail::fake() 後方可使用。

其中值得注意的是,Mail::assertSent()還有更細緻的測試方式:

Mail::assertSent(MailableClass::class, function ($mail) use ($user) {
    return $mail->hasTo($user->email) &&
           $mail->hasCc('...') &&
           $mail->hasBcc('...') &&
           $mail->hasReplyTo('...') &&
           $mail->hasFrom('...') &&
           $mail->hasSubject('...');
});
  • $mail→hasTo:驗證收件人。
  • $mail→hasCc:驗證副本收件人。
  • $mail→hasBcc:驗證密件副本收件人。
  • $mail→hasReplyTo:驗證該信是否為回覆給指定收件人之回信。
  • $mail->hasFrom:驗證寄件人。
  • $mail->hasSubject:驗證信件主旨。

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

範例

  • 測試目標:忘記密碼信索取端點

    • 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');
                  $table->string('forget_password_token', 128)->nullable();
                  $table->rememberToken();
                  $table->timestamps();
              });
          }
      
          /**
           * Reverse the migrations.
           *
           * @return void
           */
          public function down()
          {
              Schema::dropIfExists('users');
          }
      };
      
    • app/Models/User.php

      <?php
      
      namespace App\Models;
      
      use Illuminate\Database\Eloquent\Factories\HasFactory;
      use Illuminate\Foundation\Auth\User as Authenticatable;
      use Illuminate\Notifications\Notifiable;
      use Laravel\Sanctum\HasApiTokens;
      
      class User extends Authenticatable
      {
          use HasApiTokens, HasFactory, Notifiable;
      
          /**
           * The attributes that are mass assignable.
           *
           * @var array<int, string>
           */
          protected $fillable = [
              'name',
              'email',
              'password',
              'forget_password_token',
          ];
      
          /**
           * The attributes that should be hidden for serialization.
           *
           * @var array<int, string>
           */
          protected $hidden = [
              'password',
              'remember_token',
          ];
      
          /**
           * The attributes that should be cast.
           *
           * @var array<string, string>
           */
          protected $casts = [
              'email_verified_at' => 'datetime',
          ];
      }
      
    • database/factories/UserFactory.php

      <?php
      
      namespace Database\Factories;
      
      use Illuminate\Support\Str;
      use Illuminate\Database\Eloquent\Factories\Factory;
      
      class UserFactory extends Factory
      {
          /**
          * Define the model's default state.
          *
          * @return array
          */
          public function definition(): array
          {
              return [
                  'name' => $this->faker->name,
                  'email' => $this->faker->safeEmail,
                  'email_verified_at' => $this->faker->dateTime(),
                  'password' => bcrypt($this->faker->password),
                  'remember_token' => Str::random(10),
                  'forget_password_token' => Str::random(128),
              ];
          }
      }
      
    • app/Mail/ForgetPasswordMail.php

      <?php
      
      namespace App\Mail;
      
      use App\Models\User;
      use Illuminate\Bus\Queueable;
      use Illuminate\Contracts\Queue\ShouldQueue;
      use Illuminate\Mail\Mailable;
      use Illuminate\Queue\SerializesModels;
      
      class ForgetPasswordMail extends Mailable
      {
          use Queueable, SerializesModels;
      
          private $user;
      
          /**
           * Create a new message instance.
           *
           * @return void
           */
          public function __construct(User $user)
          {
              $this->user = $user;
          }
      
          /**
           * Build the message.
           *
           * @return $this
           */
          public function build()
          {
              $resetLink = config('app.url')
                  . "/password-reset?forget_password_token={$this->user->forget_password_token}";
      
              return $this
                  ->subject('To reset password')
                  ->from('example@example.com', 'Example')
                  ->with(['reset_link' => $resetLink])
                  ->view('mail.forget-password-mail');
          }
      }
      
    • resources/views/mail/forget-password-mail.blade.php

      <!DOCTYPE html>
      <html >
      <head>
          <meta charset="utf-8">
          <meta name="viewport" content="width=device-width, initial-scale=1">
          <title>Password Reset Mail</title>
      </head>
      <body class="antialiased">
          <a href="{{ $reset_link  }}">Reset Password</a>
      </body>
      </html>
      
    • routes\api.php

      <?php
      
      use App\Models\User;
      use Illuminate\Http\Request;
      use Illuminate\Support\Facades\Mail;
      use Illuminate\Support\Facades\Route;
      use App\Mail\ForgetPasswordMail;
      
      Route::post('/forget-password-mail', function (Request $request) {
          $email = $request->input('email');
      
          $user = User::where('email', '=', $email)->first();
      
          if (empty($user)) {
              return response()->json([], 404);
          }
      
          $user->forget_password_token = Str::random(128);
          $user->save();
      
          Mail::to($email)
              ->cc('cc@test.test')
              ->bcc('bcc@test.test')
              ->send(new ForgetPasswordMail($user));
      
          return response('', 200);
      })->name('retrieve-forget-password-mail');
      
  • 測試程式碼:

    <?php
    
    namespace Tests\Feature;
    
    use App\Mail\ForgetPasswordMail;
    use App\Models\User;
    use Illuminate\Foundation\Testing\RefreshDatabase;
    use Illuminate\Support\Facades\Mail;
    use Tests\TestCase;
    
    class MailTest extends TestCase
    {
        use RefreshDatabase;
    
        public function testRetriveForgetPasswordMailSuccess()
        {
            $user = User::factory()->create();
    
            $data = [
                'email' => $user->email,
            ];
    
            Mail::fake();
    
            $response = $this->post(route('retrieve-forget-password-mail'), $data);
            $response->assertOk();
    
            Mail::assertSent(ForgetPasswordMail::class, function ($mail) use ($user) {
                return $mail->hasTo($user->email);
            });
        }
    
        public function testRetriveForgetPasswordMailFailed()
        {
            $user = User::factory()->create();
    
            $data = [
                'email' => $user->email . 'x',
            ];
    
            Mail::fake();
    
            $response = $this->post(route('retrieve-forget-password-mail'), $data);
            $response->assertNotFound();
    
            Mail::assertNotSent(ForgetPasswordMail::class);
        }
    }
    

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

    • testRetriveForgetPasswordMailSuccess():在這個測試案例函數中,我們驗證了當忘記密碼信索取端點被請求時,若使用者帳號確實存在時,是否會觸發忘記密碼信 ForgotPasswordMail 寄出。
    • testRetriveForgetPasswordMailFailed():在這個測試案例函數中,我們驗證了當忘記密碼信索取端點被請求時,若使用者帳號不存在時,是否不會觸發忘記密碼信 ForgotPasswordMail 寄出。

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

明天讓我們來看看 Queue Mocking。

PS1.眼尖的讀者應該會發現,今天的內容和前一天有87%像…XD

PS2.不過還是有些細節不一樣啦

參考資料


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

尚未有邦友留言

立即登入留言