Creating and Sending Notifications

Installation

Current web applications use many different channels to send messages to the users (e.g. SMS, Slack messages, emails, push notifications, etc.). The Notifier component in Symfony is an abstraction on top of all these channels. It provides a dynamic way to manage how the messages are sent. Get the Notifier installed using:

$ composer require symfony/notifier

Channels: Chatters, Texters, Email, Browser and Push

The notifier component can send notifications to different channels. Each channel can integrate with different providers (e.g. Slack or Twilio SMS) by using transports.

The notifier component supports the following channels:

Tip

Use secrets to securely store your API’s tokens.

SMS Channel

Caution

If any of the DSN values contains any character considered special in a URI (such as +, @, $, #, /, :, *, !), you must encode them. See RFC 3986 for the full list of reserved characters or use the :phpfunction:`urlencode` function to encode them.

The SMS channel uses Symfony\Component\Notifier\Texter classes to send SMS messages to mobile phones. This feature requires subscribing to a third-party service that sends SMS messages. Symfony provides integration with a couple popular SMS services:

Service

Package

DSN

46elks

symfony/forty-six-elks-notifier

forty-six-elks://API_USERNAME:API_PASSWORD@default?from=FROM

AllMySms

symfony/all-my-sms-notifier

allmysms://LOGIN:APIKEY@default?from=FROM

AmazonSns

symfony/amazon-sns-notifier

sns://ACCESS_KEY:SECRET_KEY@default?region=REGION

Clickatell

symfony/clickatell-notifier

clickatell://ACCESS_TOKEN@default?from=FROM

ContactEveryone

symfony/contact-everyone-notifier

contact-everyone://TOKEN@default?&diffusionname=DIFFUSION_NAME&category=CATEGORY

Esendex

symfony/esendex-notifier

esendex://USER_NAME:PASSWORD@default?accountreference=ACCOUNT_REFERENCE&from=FROM

FakeSms

symfony/fake-sms-notifier

fakesms+email://MAILER_SERVICE_ID?to=TO&from=FROM or fakesms+logger://default

FreeMobile

symfony/free-mobile-notifier

freemobile://LOGIN:PASSWORD@default?phone=PHONE

GatewayApi

symfony/gateway-api-notifier

gatewayapi://TOKEN@default?from=FROM

Infobip

symfony/infobip-notifier

infobip://AUTH_TOKEN@HOST?from=FROM

Iqsms

symfony/iqsms-notifier

iqsms://LOGIN:PASSWORD@default?from=FROM

KazInfoTeh

symfony/kaz-info-teh-notifier

kaz-info-teh://USERNAME:PASSWORD@default?sender=FROM

LightSms

symfony/light-sms-notifier

lightsms://LOGIN:TOKEN@default?from=PHONE

Mailjet

symfony/mailjet-notifier

mailjet://TOKEN@default?from=FROM

MessageBird

symfony/message-bird-notifier

messagebird://TOKEN@default?from=FROM

MessageMedia

symfony/message-media-notifier

messagemedia://API_KEY:API_SECRET@default?from=FROM

Mobyt

symfony/mobyt-notifier

mobyt://USER_KEY:ACCESS_TOKEN@default?from=FROM

Nexmo

symfony/nexmo-notifier

Abandoned in favor of Vonage (symfony/vonage-notifier).

Octopush

symfony/octopush-notifier

octopush://USERLOGIN:APIKEY@default?from=FROM&type=TYPE

OrangeSms

symfony/orange-sms-notifier

orange-sms://CLIENT_ID:CLIENT_SECRET@default?from=FROM&sender_name=SENDER_NAME

OvhCloud

symfony/ovh-cloud-notifier

ovhcloud://APPLICATION_KEY:APPLICATION_SECRET@default?consumer_key=CONSUMER_KEY&service_name=SERVICE_NAME&no_stop_clause=true

Sendberry

symfony/sendberry-notifier

sendberry://USERNAME:PASSWORD@default?auth_key=AUTH_KEY&from=FROM

Sendinblue

symfony/sendinblue-notifier

sendinblue://API_KEY@default?sender=PHONE

Sms77

symfony/sms77-notifier

sms77://API_KEY@default?from=FROM

Sinch

symfony/sinch-notifier

sinch://ACCOUNT_ID:AUTH_TOKEN@default?from=FROM

Smsapi

symfony/smsapi-notifier

smsapi://TOKEN@default?from=FROM&test=0

SmsBiuras

symfony/sms-biuras-notifier

smsbiuras://UID:API_KEY@default?from=FROM&test_mode=0

Smsc

symfony/smsc-notifier

smsc://LOGIN:PASSWORD@default?from=FROM

SMSFactor

symfony/sms-factor-notifier

sms-factor://TOKEN@default?sender=SENDER&push_type=PUSH_TYPE

SpotHit

symfony/spot-hit-notifier

spothit://TOKEN@default?from=FROM

Telnyx

symfony/telnyx-notifier

telnyx://API_KEY@default?from=FROM&messaging_profile_id=MESSAGING_PROFILE_ID

TurboSms

symfony/turbo-sms-notifier

turbosms://AUTH_TOKEN@default?from=FROM

Twilio

symfony/twilio-notifier

twilio://SID:TOKEN@default?from=FROM

Vonage

symfony/vonage-notifier

vonage://KEY:SECRET@default?from=FROM

Yunpian

symfony/yunpian-notifier

yunpian://APIKEY@default

New in version 6.1: The 46elks, OrangeSms, KazInfoTeh and Sendberry integrations were introduced in Symfony 6.1. The no_stop_clause option in OvhCloud DSN was introduced in Symfony 6.1. The test option in Smsapi DSN was introduced in Symfony 6.1.

New in version 6.2: The ContactEveryone and SMSFactor integrations were introduced in Symfony 6.2.

To enable a texter, add the correct DSN in your .env file and configure the texter_transports:

# .env
TWILIO_DSN=twilio://SID:TOKEN@default?from=FROM

Chat Channel

Caution

If any of the DSN values contains any character considered special in a URI (such as +, @, $, #, /, :, *, !), you must encode them. See RFC 3986 for the full list of reserved characters or use the :phpfunction:`urlencode` function to encode them.

The chat channel is used to send chat messages to users by using Symfony\Component\Notifier\Chatter classes. Symfony provides integration with these chat services:

Service

Package

DSN

AmazonSns

symfony/amazon-sns-notifier

sns://ACCESS_KEY:SECRET_KEY@default?region=REGION

Discord

symfony/discord-notifier

discord://TOKEN@default?webhook_id=ID

FakeChat

symfony/fake-chat-notifier

fakechat+email://default?to=TO&from=FROM or fakechat+logger://default

Firebase

symfony/firebase-notifier

firebase://USERNAME:PASSWORD@default

Gitter

symfony/gitter-notifier

gitter://TOKEN@default?room_id=ROOM_ID

GoogleChat

symfony/google-chat-notifier

googlechat://ACCESS_KEY:ACCESS_TOKEN@default/SPACE?thread_key=THREAD_KEY

LinkedIn

symfony/linked-in-notifier

linkedin://TOKEN:USER_ID@default

Mattermost

symfony/mattermost-notifier

mattermost://ACCESS_TOKEN@HOST/PATH?channel=CHANNEL

Mercure

symfony/mercure-notifier

mercure://HUB_ID?topic=TOPIC

MicrosoftTeams

symfony/microsoft-teams-notifier

microsoftteams://default/PATH

RocketChat

symfony/rocket-chat-notifier

rocketchat://TOKEN@ENDPOINT?channel=CHANNEL

Slack

symfony/slack-notifier

slack://TOKEN@default?channel=CHANNEL

Telegram

symfony/telegram-notifier

telegram://TOKEN@default?channel=CHAT_ID

Zendesk

symfony/zendesk-notifier

zendesk://EMAIL:TOKEN@SUBDOMAIN

Zulip

symfony/zulip-notifier

zulip://EMAIL:TOKEN@HOST?channel=CHANNEL

New in version 6.2: The Zendesk integration was introduced in Symfony 6.2.

Chatters are configured using the chatter_transports setting:

# .env
SLACK_DSN=slack://TOKEN@default?channel=CHANNEL

Email Channel

The email channel uses the Symfony Mailer to send notifications using the special Symfony\Bridge\Twig\Mime\NotificationEmail. It is required to install the Twig bridge along with the Inky and CSS Inliner Twig extensions:

$ composer require symfony/twig-pack twig/cssinliner-extra twig/inky-extra

After this, configure the mailer. You can also set the default “from” email address that should be used to send the notification emails:

Push Channel

Caution

If any of the DSN values contains any character considered special in a URI (such as +, @, $, #, /, :, *, !), you must encode them. See RFC 3986 for the full list of reserved characters or use the :phpfunction:`urlencode` function to encode them.

The push channel is used to send notifications to users by using Symfony\Component\Notifier\Texter classes. Symfony provides integration with these push services:

Service

Package

DSN

Engagespot

symfony/engagespot-notifier

engagespot://API_KEY@default?campaign_name=CAMPAIGN_NAME

Expo

symfony/expo-notifier

expo://Token@default

OneSignal

symfony/one-signal-notifier

onesignal://APP_ID:API_KEY@default?defaultRecipientId=DEFAULT_RECIPIENT_ID''

New in version 6.1: The Engagespot integration was introduced in Symfony 6.1.

To enable a texter, add the correct DSN in your .env file and configure the texter_transports:

# .env
EXPO_DSN=expo://TOKEN@default

Configure to use Failover or Round-Robin Transports

Besides configuring one or more separate transports, you can also use the special || and && characters to implement a failover or round-robin transport:

Creating & Sending Notifications

To send a notification, autowire the Symfony\Component\Notifier\NotifierInterface (service ID notifier). This class has a send() method that allows you to send a Symfony\Component\Notifier\Notification\Notification to a Symfony\Component\Notifier\Recipient\Recipient:

// src/Controller/InvoiceController.php
namespace App\Controller;

use Symfony\Component\Notifier\Notification\Notification;
use Symfony\Component\Notifier\NotifierInterface;
use Symfony\Component\Notifier\Recipient\Recipient;

class InvoiceController extends AbstractController
{
    #[Route('/invoice/create')]
    public function create(NotifierInterface $notifier)
    {
        // ...

        // Create a Notification that has to be sent
        // using the "email" channel
        $notification = (new Notification('New Invoice', ['email']))
            ->content('You got a new invoice for 15 EUR.');

        // The receiver of the Notification
        $recipient = new Recipient(
            $user->getEmail(),
            $user->getPhonenumber()
        );

        // Send the notification to the recipient
        $notifier->send($notification, $recipient);

        // ...
    }
}

The Notification is created by using two arguments: the subject and channels. The channels specify which channel (or transport) should be used to send the notification. For instance, ['email', 'sms'] will send both an email and sms notification to the user.

The default notification also has a content() and emoji() method to set the notification content and icon.

Symfony provides the following recipients:

Symfony\Component\Notifier\Recipient\NoRecipient

This is the default and is useful when there is no need to have information about the receiver. For example, the browser channel uses the current requests’ session flashbag;

Symfony\Component\Notifier\Recipient\Recipient

This can contain both the email address and the phone number of the user. This recipient can be used for all channels (depending on whether they are actually set).

Configuring Channel Policies

Instead of specifying the target channels on creation, Symfony also allows you to use notification importance levels. Update the configuration to specify what channels should be used for specific levels (using channel_policy):

Now, whenever the notification’s importance is set to “high”, it will be sent using the Slack transport:

// ...
class InvoiceController extends AbstractController
{
    #[Route('/invoice/create')]
    public function invoice(NotifierInterface $notifier)
    {
        // ...

        $notification = (new Notification('New Invoice'))
            ->content('You got a new invoice for 15 EUR.')
            ->importance(Notification::IMPORTANCE_HIGH);

        $notifier->send($notification, new Recipient('wouter@example.com'));

        // ...
    }
}

Customize Notifications

You can extend the Notification or Recipient base classes to customize their behavior. For instance, you can overwrite the getChannels() method to only return sms if the invoice price is very high and the recipient has a phone number:

namespace App\Notifier;

use Symfony\Component\Notifier\Notification\Notification;
use Symfony\Component\Notifier\Recipient\RecipientInterface;
use Symfony\Component\Notifier\Recipient\SmsRecipientInterface;

class InvoiceNotification extends Notification
{
    private $price;

    public function __construct(int $price)
    {
        $this->price = $price;
    }

    public function getChannels(RecipientInterface $recipient)
    {
        if (
            $this->price > 10000
            && $recipient instanceof SmsRecipientInterface
        ) {
            return ['sms'];
        }

        return ['email'];
    }
}

Customize Notification Messages

Each channel has its own notification interface that you can implement to customize the notification message. For instance, if you want to modify the message based on the chat service, implement Symfony\Component\Notifier\Notification\ChatNotificationInterface and its asChatMessage() method:

// src/Notifier/InvoiceNotification.php
namespace App\Notifier;

use Symfony\Component\Notifier\Message\ChatMessage;
use Symfony\Component\Notifier\Notification\ChatNotificationInterface;
use Symfony\Component\Notifier\Notification\Notification;
use Symfony\Component\Notifier\Recipient\RecipientInterface;

class InvoiceNotification extends Notification implements ChatNotificationInterface
{
    private $price;

    public function __construct(int $price)
    {
        $this->price = $price;
    }

    public function asChatMessage(RecipientInterface $recipient, string $transport = null): ?ChatMessage
    {
        // Add a custom subject and emoji if the message is sent to Slack
        if ('slack' === $transport) {
            $this->subject('You\'re invoiced '.strval($this->price).' EUR.');
            $this->emoji("money");
            return ChatMessage::fromNotification($this);
        }

        // If you return null, the Notifier will create the ChatMessage
        // based on this notification as it would without this method.
        return null;
    }
}

The Symfony\Component\Notifier\Notification\SmsNotificationInterface and Symfony\Component\Notifier\Notification\EmailNotificationInterface also exists to modify messages sent to those channels.

Customize Browser Notifications (Flash Messages)

New in version 6.1: Support for customizing importance levels was introduced in Symfony 6.1.

The default behavior for browser channel notifications is to add a flash message with notification as its key.

However, you might prefer to map the importance level of the notification to the type of flash message, so you can tweak their style.

you can do that by overriding the default notifier.flash_message_importance_mapper service with your own implementation of Symfony\Component\Notifier\FlashMessage\FlashMessageImportanceMapperInterface where you can provide your own “importance” to “alert level” mapping.

Symfony currently provides an implementation for the Bootstrap CSS framework’s typical alert levels, which you can implement immediately using:

Disabling Delivery

While developing (or testing), you may want to disable delivery of notifications entirely. You can do this by forcing Notifier to use the NullTransport for all configured texter and chatter transports only in the dev (and/or test) environment:

# config/packages/dev/notifier.yaml
framework:
    notifier:
        texter_transports:
            twilio: 'null://null'
        chatter_transports:
            slack: 'null://null'

Learn more