108
108

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

ゆめみAdvent Calendar 2019

Day 19

Laravel Notification をやさしく解説する

Last updated at Posted at 2019-12-19

これはなに?

Laravel 6.x で通知を実装するときのお作法を解説します。
全ては ドキュメント に書いてある通りなんですが、ちょっとわかりにくいと思いました。
僕は一読しただけでは理解できませんでした。

なのでこの記事では、自分なりに噛み砕いた表現で Laravel Notification を説明します。
ドキュメントをぱっと読んでみたけどわからんかった、という人の理解の助けになれば幸いです。

解説すること

  • 基本的な通知の実装方法
  • カスタム通知チャネルの実装方法
    • AWS SDK を利用したサンプル

解説しないこと

  • 通知方法ごとの通信処理の設定方法
    • 例えば、メール送信のそのものの設定方法など
  • 通知イベントの取り扱い
  • キューとワーカーを用いた通知の非同期処理

対象読者

  • Laravel で通知を初めて実装する人
  • 通知のドキュメントを読んだけどよく分からなった人

読まなくてもいい人

  • 通知のドキュメントの内容をおおよそ理解できている人

Notifications - Laravel

通知の三大要素

Laravel の通知には大きく 3 つの要素があります。

  1. 通知メッセージ
  2. 通知方法
  3. 通知先

image.png

実装では、これらの要素を Class や Trait ( を use した Class ) として作っていきます。

  1. Notification Class
  2. Channel Class
  3. Notifiable Class ( used Notifiable Trait )

image.png

以降では、ユーザー ( 通知先 )予約を申し込んだこと ( 通知メッセージ )メール ( 通知方法 ) で通知する、というシナリオに沿ってサンプルを提示しながら解説していきます。

image.png

通知メッセージ ( Notification )

  • Notification Class
  • php artisan make:notificationapp/Notifications 配下に作成される

通知メッセージは以下の機能を持ちます。

  • 通知する内容 ( メッセージ ) を生成
  • 通知方法を選択

通知する内容 ( メッセージ ) を生成

  • 例えば、メール通知ならメール本文
  • 例えば、 Slack 通知なら Slack メッセージ本文

通知内容を生成する流れとしては、以下がお決まりのパターンです。

1. コンストラクタで通知内容を生成する情報源を受け取る

app/Notifications/Reserved.php
public function __construct(Reservation $reservation)
{
    $this->reservation = $reservation;
}

2. 通知方法に応じた通知内容を返すメソッドを実装する

  • 例えば、メール通知なら toMail() に実装
  • 例えば、 Slack 通知なら toSlack() に実装

参考: Formatting Mail Messages

app/Notifications/Reserved.php
public function toSlack($notifiable)
{
    return (new MailMessage)
        ->greeting('予約を受け付けました')
        ->line('予約番号:' . $this->reservation->id)
        ->line('予約日:' . $this->reservation->date)
        ->line('氏名:' . $notifiable->name)
        ->line('当日のご来店をお待ちしております。');
}

$notifiable は、後ほど解説する通知先のインスタンスです。
上記のサンプルコードは予約をした文脈なので、予約を申し込んだユーザーが $notifiable に入ります。

通知方法を選択

参考: Specifying Delivery Channels

via() メソッドで通知方法を選択します。
一つの通知方法しかないなら決め打ちで良いです。

app/Notifications/Reserved.php
public function via($notifiable)
{
    return ['mail'];
}

環境によって切り替えたいというのは、よくあるニーズかもしれません。
例えば、ローカル環境ではテストのため Slack に通知を飛ばすなど。

app/Notifications/Reserved.php
public function via($notifiable)
{
    return \App::environment('local') ? ['slack'] : ['mail'];
}

ユーザーの属性によって通知先を変えることもあるでしょう。
例えば、とあるユーザーはメールではなく SMS で通知を受けたいなど。

app/Notifications/Reserved.php
public function via($notifiable)
{
    return $notifiable->prefers_sms ? ['nexmo'] : ['mail'];
}

Notification class の使い方

実際に通知を送る処理を書くときには、実装した Notification class をインスタンス化して利用します。

use App\Notifications\Reserved;

$notification = new Reserved($reservation);

通知先 ( Notifiable )

  • 通知先となるインスタンス
  • 例えば、今回の解説では予約を申し込んだ ユーザー

通知先は以下の機能を持ちます。

  • 通知送信メソッド
  • 通知に伴う通信先情報の提供

通知送信メソッド

実装はとても簡単で、通知先にしたいクラスで Notifiable trait を use するだけです。

app/User.php
<?php

namespace App;

use Illuminate\Foundation\Auth\User as Authenticatable;
use Illuminate\Notifications\Notifiable;

class User extends Authenticatable
{
    use Notifiable;
}

参考: Using The Notifiable Trait

これだけで、ユーザーに通知を送れるようになります。

use App\Notifications\Reserved;

$user->notify(new Reserved($reservation));

複数のユーザーに同一の通知を送りたい場合は、 Notification ファサードの send() メソッドが便利です。

Notification::send($users, new Reserved($reservation));

$usersNotifiable インスタンスの配列もしくはコレクションです。

通知方法に応じた通信先情報の提供

通知方法としてメールを送信する場合、送信先のメールアドレスが必要です。
デフォルトでは $notifiable->email を参照しますが、これを変更したい場合は以下のメソッドを実装します。

app/User.php
public function routeNotificationForMail($notification)
{
    return $this->email_address;
}

この routeNotificationForMail というメソッド名は、通知方法 ( Channel ) クラスによる規約です。
他の通知方法の場合は、それぞれの規約に応じたメソッドを実装する必要があります。

  • Slack: Webhook URL を返す
  • SMS: 電話番号を返す

など。詳しくはドキュメントを参照ください。

通知方法 ( Channel )

  • 通知のプロトコル
  • メールや Slack, SMS などは Laravel に組み込みで実装されている
  • デフォルトの通知方法を利用する場合は、ここはノータッチで OK
  • カスタムチャネルを作ろうと思っている人は、一度フレームワークの実装を見てみてもいいかも

laravel/framework/src/Illuminate/Notifications/Channels/MailChannel.php

通知方法は以下の機能を持ちます。

プロトコルに応じた通知先との通信処理 1

$notifiable と Notification class を受け取る send() メソッドを実装します。

参考: Custom Channels

今回は AWS SDK を利用して AWS SNS 経由で SMS を送るカスタムチャネルを作ってみます。
Channel class のファイルの場所の規約はないので、好きなディレクトリ配下に置いて構いません。
サンプルでは app/Notifications/Channels というディレクトリ配下に作成しました。

app/Notifications/AwsSnsPublishSms.php
<?php

namespace App\Notifications\Channels;

use Illuminate\Support\Facades\App;
use App\Notifications\Reserved;

class AwsSnsPublishSms
{
    /**
     * 電話番号への SMS 通知の送信
     *
     * @param  $notifiable
     * @param  \App\Notifications\Reserved $notification
     * @return void
     */
    public function send($notifiable, Reserved $notification)
    {
        // SMS 用の通知メッセージを返すメソッド toPbulishSms() の実装を Notification class の規約とした
        $message = $notification->toPublishSms($notifiable);

        $snsClient = App::make('aws')->createClient('Sns');
        $snsClient->publish([
            'Message'     => $message,
            'PhoneNumber' => $this->toE164nizeInJapan($notifiable->phone_number),
        ]);
    }

    /**
     * AWS SNS で SMS を送信するためには E.164 形式の電話番号が必要
     * 
     * @return string E.164nize phone number in JP
     */
    private function toE164nizeInJapan($phone_numeber) : string
    {
        return '+81' . substr($phone_numeber, 1);
    }
}
app/Notifications/Reserved.php
public function toPbulishSms($notifiable)
{
    // メッセージの整形は blade テンプレートを使うとスッキリ書けます
    return view('otifications.reserved', [
            'id'   => $this->reservation->id,
            'date' => $this->reservation->date,
            'name' => $notifiable->name,
        ])->render();
}
resources/views/notifications/reserved.blade.php
予約を受け付けました
予約番号: {{ $id }}
予約日: {{ $date }}
氏名: {{ $notifiable }}
当日のご来店をお待ちしております

あとは、 Notification class の via() メソッドでカスタムチャネルを指定すれば完了です。

app/Notifications/Reserved.php
use App\Notifications\Channels\AwsSnsPublishSms;

public function via($notifiable)
{
    return [AwsSnsPublishSms::class];
}

サードパーティのカスタムチャネル

Laravel Notification Channels というサイトにサードパーティのカスタムチャネルがのっています。

  • APN などプラットフォームの規格
  • Twitter, Facebook 等の SNS / メッセンジャーサービス
  • Twillo などの SMS や VoIP サービス

などが対応しています。
利用したい通知先のサービスやプロトコルがあるなら、まずはここから探してみてもいいかもしれません。
僕が欲しかった AWS SNS は残念ながらなかったので、泣く泣くカスタムチャネルを実装しました。 2

まとめ

  • Laravel の通知の実装には大きく 3 つの要素がある
  • 通知メッセージ ( Notification class ) でやること
    • 通知内容
    • 通知方法の選択
  • 通知方法 ( Channel class ) でやること
    • プロトコルに応じた通知の通信処理
  • 通知先 ( Notifiable ) でやること
    • 通知送信メソッドの実装
      • Notifiable trait を use するだけで良い
    • 通知方法に応じた通信先情報の提供

Laravel での通知の実装方法がよくわかってなくて手が出せてなかった方も、ぜひ一度、通知処理を実装してみてください。
今回取り上げませんでしたが、 Slack 通知が incoming-webhook でとってもお手軽に試せるので、オススメです。

参考

  1. DB Notification など、通信っぽくないものありますが、アプリケーションの外とやり取りするという意味で通信という言葉で表現しています

  2. 昔あったみたいだけど、メンテナンスされてなくて消えたっぽい

108
108
1

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
108
108

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?