Notificaciones
- Introducción
- Generando Notificaciones
- Enviando Notificaciones
-
Notificaciones por Mail
- Formateando Mensajes de Mail
- Personalizando el Remitente
- Personalizando el Destinatario
- Personalizando el Asunto
- Personalizando el Mailer
- Personalizando las Plantillas
- Adjuntos
- Añadiendo Etiquetas y Metadatos
- Personalizando el Mensaje de Symfony
- Usando Mailables
- Previsualizando Notificaciones por Mail
- Notificaciones por Mail Markdown
- Notificaciones de Base de Datos
- Notificaciones de Difusión
- Notificaciones SMS
- Notificaciones de Slack
- Localizando Notificaciones
- Pruebas
- Eventos de Notificación
- Canales Personalizados
Introducción
Además del soporte para enviar correos electrónicos, Laravel ofrece soporte para enviar notificaciones a través de una variedad de canales de entrega, incluyendo correo electrónico, SMS (a través de Vonage, anteriormente conocido como Nexmo), y Slack. Además, se han creado una variedad de canales de notificación construidos por la comunidad para enviar notificaciones a través de docenas de canales diferentes. ¡Las notificaciones también se pueden almacenar en una base de datos para que puedan mostrarse en tu interfaz web! Típicamente, las notificaciones deben ser mensajes cortos e informativos que notifiquen a los usuarios sobre algo que ocurrió en su aplicación. Por ejemplo, si estás escribiendo una aplicación de facturación, podrías enviar una notificación de "Factura Pagada" a tus usuarios a través de los canales de correo electrónico y SMS.
Generando Notificaciones
En Laravel, cada notificación está representada por una sola clase que se almacena típicamente en el directorio app/Notifications
. No te preocupes si no ves este directorio en tu aplicación: se creará para ti cuando ejecutes el comando Artisan make:notification
:
php artisan make:notification InvoicePaid
Este comando colocará una nueva clase de notificación en tu directorio app/Notifications
. Cada clase de notificación contiene un método via
y un número variable de métodos de construcción de mensajes, como toMail
o toDatabase
, que convierten la notificación en un mensaje adaptado para ese canal particular.
Enviando Notificaciones
Usando el Trait Notifiable
Las notificaciones pueden enviarse de dos maneras: utilizando el método notify
del trait Notifiable
o utilizando la Notification
facade. El trait Notifiable
está incluido en el modelo App\Models\User
de tu aplicación por defecto:
<?php namespace App\Models; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; class User extends Authenticatable { use Notifiable; }
El método notify
que proporciona este trait espera recibir una instancia de notificación:
use App\Notifications\InvoicePaid; $user->notify(new InvoicePaid($invoice));
Recuerda que puedes usar el trait Notifiable
en cualquiera de tus modelos. No estás limitado a incluirlo solo en tu modelo User
.
Usando la Facade de Notificación
Alternativamente, puedes enviar notificaciones a través de la facade
Notification. Este enfoque es útil cuando necesitas enviar una notificación a múltiples entidades notificables, como una colección de usuarios. Para enviar notificaciones utilizando la facade, pasa todas las entidades notificables y la instancia de notificación al método send
:
use Illuminate\Support\Facades\Notification; Notification::send($users, new InvoicePaid($invoice));
También puedes enviar notificaciones de inmediato utilizando el método sendNow
. Este método enviará la notificación de inmediato incluso si la notificación implementa la interfaz ShouldQueue
:
Notification::sendNow($developers, new DeploymentCompleted($deployment));
Especificando Canales de Entrega
Cada clase de notificación tiene un método via
que determina en qué canales se entregará la notificación. Las notificaciones pueden enviarse a través de los canales mail
, database
, broadcast
, vonage
y slack
.
Si deseas utilizar otros canales de entrega como Telegram o Pusher, visita el sitio web de Laravel Notification Channels impulsado por la comunidad.
El método via
recibe una instancia de $notifiable
, que será una instancia de la clase a la que se está enviando la notificación. Puedes usar $notifiable
para determinar en qué canales debe entregarse la notificación:
/** * Get the notification's delivery channels. * * @return array<int, string> */ public function via(object $notifiable): array { return $notifiable->prefers_sms ? ['vonage'] : ['mail', 'database']; }
Encolando Notificaciones
Antes de encolar notificaciones, debes configurar tu cola y iniciar un worker.
Enviar notificaciones puede llevar tiempo, especialmente si el canal necesita hacer una llamada a una API externa para entregar la notificación. Para acelerar el tiempo de respuesta de tu aplicación, permite que tu notificación sea encolada agregando la interfaz ShouldQueue
y el trait Queueable
a tu clase. La interfaz y el trait ya están importados para todas las notificaciones generadas utilizando el comando make:notification
, así que puedes añadirlos inmediatamente a tu clase de notificación:
<?php namespace App\Notifications; use Illuminate\Bus\Queueable; use Illuminate\Contracts\Queue\ShouldQueue; use Illuminate\Notifications\Notification; class InvoicePaid extends Notification implements ShouldQueue { use Queueable; // ... }
Una vez que se haya añadido la interfaz ShouldQueue
a tu notificación, puedes enviar la notificación como de costumbre. Laravel detectará la interfaz ShouldQueue
en la clase y, automáticamente, encolará la entrega de la notificación:
Al hacer cola de notificaciones, se creará un trabajo en cola para cada combinación de destinatario y canal. Por ejemplo, se despacharán seis trabajos a la cola si tu notificación tiene tres destinatarios y dos canales.
Retraso de Notificaciones
Si deseas retrasar la entrega de la notificación, puedes encadenar el método delay
a la instancia de tu notificación:
$delay = now()->addMinutes(10); $user->notify((new InvoicePaid($invoice))->delay($delay));
Puedes pasar un array al método delay
para especificar la cantidad de demora para canales específicos:
$user->notify((new InvoicePaid($invoice))->delay([ 'mail' => now()->addMinutes(5), 'sms' => now()->addMinutes(10), ]));
Alternativamente, puedes definir un método withDelay
en la propia clase de notificación. El método withDelay
debe devolver un array de nombres de canales y valores de retraso:
/** * Determine the notification's delivery delay. * * @return array<string, \Illuminate\Support\Carbon> */ public function withDelay(object $notifiable): array { return [ 'mail' => now()->addMinutes(5), 'sms' => now()->addMinutes(10), ]; }
Personalizando la Conexión de la Cola de Notificaciones
Por defecto, las notificaciones en cola se encolarán utilizando la conexión de cola predeterminada de tu aplicación. Si deseas especificar una conexión diferente que se debe usar para una notificación en particular, puedes llamar al método onConnection
desde el constructor de tu notificación:
<?php namespace App\Notifications; use Illuminate\Bus\Queueable; use Illuminate\Contracts\Queue\ShouldQueue; use Illuminate\Notifications\Notification; class InvoicePaid extends Notification implements ShouldQueue { use Queueable; /** * Create a new notification instance. */ public function __construct() { $this->onConnection('redis'); } }
O, si deseas especificar una conexión de cola específica que se debe usar para cada canal de notificación admitido por la notificación, puedes definir un método viaConnections
en tu notificación. Este método debe devolver un array de pares de nombre de canal / nombre de conexión de cola:
/** * Determine which connections should be used for each notification channel. * * @return array<string, string> */ public function viaConnections(): array { return [ 'mail' => 'redis', 'database' => 'sync', ]; }
Personalizando las Colas del Canal de Notificación
Si deseas especificar una cola específica que debe utilizarse para cada canal de notificación admitido por la notificación, puedes definir un método viaQueues
en tu notificación. Este método debe devolver un array de pares de nombre de canal / nombre de cola:
/** * Determine which queues should be used for each notification channel. * * @return array<string, string> */ public function viaQueues(): array { return [ 'mail' => 'mail-queue', 'slack' => 'slack-queue', ]; }
Middleware de Notificación en Cola
Las notificaciones en cola pueden definir middleware al igual que los trabajos en cola. Para comenzar, define un método middleware
en tu clase de notificación. El método middleware
recibirá las variables $notifiable
y $channel
, lo que te permite personalizar el middleware devuelto según el destino de la notificación:
use Illuminate\Queue\Middleware\RateLimited; /** * Get the middleware the notification job should pass through. * * @return array<int, object> */ public function middleware(object $notifiable, string $channel) { return match ($channel) { 'email' => [new RateLimited('postmark')], 'slack' => [new RateLimited('slack')], default => [], }; }
Notificaciones en Cola y Transacciones de Base de Datos
Cuando se despachan notificaciones en cola dentro de transacciones de base de datos, pueden ser procesadas por la cola antes de que la transacción de la base de datos se haya confirmado. Cuando esto sucede, cualquier actualización que hayas realizado a modelos o registros de base de datos durante la transacción de base de datos puede no reflejarse aún en la base de datos. Además, cualquier modelo o registro de base de datos creado dentro de la transacción puede no existir en la base de datos. Si tu notificación depende de estos modelos, pueden ocurrir errores inesperados cuando se procesa el trabajo que envía la notificación en cola.
Si la opción de configuración after_commit
de la conexión de cola está configurada en false
, aún puedes indicar que una notificación en cola particular debe ser despachada después de que se hayan confirmado todas las transacciones de base de datos abiertas llamando al método afterCommit
al enviar la notificación:
use App\Notifications\InvoicePaid; $user->notify((new InvoicePaid($invoice))->afterCommit());
Alternativamente, puedes llamar al método afterCommit
desde el constructor de tu notificación:
<?php namespace App\Notifications; use Illuminate\Bus\Queueable; use Illuminate\Contracts\Queue\ShouldQueue; use Illuminate\Notifications\Notification; class InvoicePaid extends Notification implements ShouldQueue { use Queueable; /** * Create a new notification instance. */ public function __construct() { $this->afterCommit(); } }
Para obtener más información sobre cómo solucionar estos problemas, revisa la documentación sobre trabajos en cola y transacciones de base de datos.
Determinando si se debe enviar una notificación en cola
Después de que se haya despachado una notificación en cola para su procesamiento en segundo plano, típicamente será aceptada por un trabajador de cola y enviada a su destinatario previsto.
Sin embargo, si deseas tomar la decisión final sobre si la notificación en cola debe ser enviada después de que esté siendo procesada por un trabajador de cola, puedes definir un método shouldSend
en la clase de notificación. Si este método devuelve false
, la notificación no será enviada:
/** * Determine if the notification should be sent. */ public function shouldSend(object $notifiable, string $channel): bool { return $this->invoice->isPaid(); }
Notificaciones Bajo Demanda
A veces es posible que necesites enviar una notificación a alguien que no está almacenado como un "usuario" de tu aplicación. Usando el método route
de la fachada Notification
, puedes especificar información de enrutamiento de notificación ad-hoc antes de enviar la notificación:
use Illuminate\Broadcasting\Channel; use Illuminate\Support\Facades\Notification; Notification::route('mail', 'taylor@example.com') ->route('vonage', '5555555555') ->route('slack', '#slack-channel') ->route('broadcast', [new Channel('channel-name')]) ->notify(new InvoicePaid($invoice));
Si deseas proporcionar el nombre del destinatario al enviar una notificación bajo demanda a la ruta mail
, puedes proporcionar un array que contenga la dirección de correo electrónico como la clave y el nombre como el valor del primer elemento del array:
Notification::route('mail', [ 'barrett@example.com' => 'Barrett Blair', ])->notify(new InvoicePaid($invoice));
Usando el método routes
, puedes proporcionar información de enrutamiento ad-hoc para múltiples canales de notificación a la vez:
Notification::routes([ 'mail' => ['barrett@example.com' => 'Barrett Blair'], 'vonage' => '5555555555', ])->notify(new InvoicePaid($invoice));
Notificaciones por Mail
Formateando Mensajes de Mail
Si una notificación admite ser enviada como un correo electrónico, debes definir un método toMail
en la clase de notificación. Este método recibirá una entidad $notifiable
y debe devolver una instancia de Illuminate\Notifications\Messages\MailMessage
.
La clase MailMessage
contiene algunos métodos simples para ayudarte a crear mensajes de correo electrónico transaccionales. Los mensajes de correo pueden contener líneas de texto así como un "llamado a la acción". Echemos un vistazo a un ejemplo del método toMail
:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { $url = url('/invoice/'.$this->invoice->id); return (new MailMessage) ->greeting('Hello!') ->line('One of your invoices has been paid!') ->lineIf($this->amount > 0, "Amount paid: {$this->amount}") ->action('View Invoice', $url) ->line('Thank you for using our application!'); }
Nota que estamos utilizando $this->invoice->id
en nuestro método toMail
. Puedes pasar cualquier dato que tu notificación necesite para generar su mensaje en el constructor de la notificación.
En este ejemplo, registramos un saludo, una línea de texto, un llamado a la acción y luego otra línea de texto. Estos métodos proporcionados por el objeto MailMessage
hacen que sea simple y rápido formatear pequeños correos electrónicos transaccionales. El canal de correo luego traducirá los componentes del mensaje en una hermosa plantilla de correo electrónico HTML receptiva con un contraparte de texto plano. Aquí hay un ejemplo de un correo electrónico generado por el canal mail
:
[!NOTA]
Al enviar notificaciones por correo, asegúrate de configurar la opción de configuración name
en tu archivo de configuración config/app.php
. Este valor se utilizará en el encabezado y pie de tus mensajes de notificación por correo.
Mensajes de Error
Algunas notificaciones informan a los usuarios sobre errores, como un pago de factura fallido. Puedes indicar que un mensaje de correo está relacionado con un error llamando al método error
al construir tu mensaje. Al usar el método error
en un mensaje de correo, el botón de llamada a la acción será rojo en lugar de negro:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->error() ->subject('Invoice Payment Failed') ->line('...'); }
Otras Opciones de Formato de Notificación por Correo
En lugar de definir las "líneas" de texto en la clase de notificación, puedes usar el método view
para especificar una plantilla personalizada que se debe utilizar para renderizar el correo electrónico de notificación:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage)->view( 'mail.invoice.paid', ['invoice' => $this->invoice] ); }
Puedes especificar una vista de texto plano para el mensaje de correo pasando el nombre de la vista como el segundo elemento de un array que se le da al método view
:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage)->view( ['mail.invoice.paid', 'mail.invoice.paid-text'], ['invoice' => $this->invoice] ); }
O, si tu mensaje solo tiene una vista de texto plano, puedes utilizar el método text
:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage)->text( 'mail.invoice.paid-text', ['invoice' => $this->invoice] ); }
Personalizando el Remitente
Por defecto, la dirección del remitente / dirección de desde del correo electrónico se define en el archivo de configuración config/mail.php
. Sin embargo, puedes especificar la dirección de desde para una notificación específica utilizando el método from
:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->from('barrett@example.com', 'Barrett Blair') ->line('...'); }
Personalizando el Destinatario
Al enviar notificaciones a través del canal mail
, el sistema de notificaciones buscará automáticamente una propiedad email
en tu entidad notifiable. Puedes personalizar qué dirección de correo electrónico se utiliza para entregar la notificación definiendo un método routeNotificationForMail
en la entidad notifiable:
<?php namespace App\Models; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; use Illuminate\Notifications\Notification; class User extends Authenticatable { use Notifiable; /** * Route notifications for the mail channel. * * @return array<string, string>|string */ public function routeNotificationForMail(Notification $notification): array|string { // Return email address only... return $this->email_address; // Return email address and name... return [$this->email_address => $this->name]; } }
Personalizando el Asunto
Por defecto, el asunto del correo electrónico es el nombre de la clase de la notificación formateado en "Title Case". Así que, si tu clase de notificación se llama InvoicePaid
, el asunto del correo electrónico será Invoice Paid
. Si deseas especificar un asunto diferente para el mensaje, puedes llamar al método subject
al construir tu mensaje:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->subject('Notification Subject') ->line('...'); }
Personalizando el Mailer
Por defecto, la notificación por correo electrónico se enviará utilizando el mailer predeterminado definido en el archivo de configuración config/mail.php
. Sin embargo, puedes especificar un mailer diferente en tiempo de ejecución llamando al método mailer
al construir tu mensaje:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->mailer('postmark') ->line('...'); }
Personalizando las Plantillas
Puedes modificar la plantilla HTML y de texto plano utilizada por las notificaciones de correo publicando los recursos del paquete de notificaciones. Después de ejecutar este comando, las plantillas de notificación de correo se ubicará en el directorio resources/views/vendor/notifications
:
php artisan vendor:publish --tag=laravel-notifications
Adjuntos
Para agregar archivos adjuntos a una notificación por correo electrónico, utiliza el método attach
mientras construyes tu mensaje. El método attach
acepta la ruta absoluta al archivo como su primer argumento:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->greeting('Hello!') ->attach('/path/to/file'); }
El método attach
que ofrecen los mensajes de correo de notificación también acepta objetos adjuntos. Por favor, consulta la completa documentación de objetos adjuntos para aprender más.
Al adjuntar archivos a un mensaje, también puedes especificar el nombre a mostrar y / o el tipo MIME pasando un array
como segundo argumento al método attach
:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->greeting('Hello!') ->attach('/path/to/file', [ 'as' => 'name.pdf', 'mime' => 'application/pdf', ]); }
A diferencia de adjuntar archivos en objetos mailable, no puedes adjuntar un archivo directamente desde un disco de almacenamiento utilizando attachFromStorage
. Debes usar el método attach
con una ruta absoluta al archivo en el disco de almacenamiento. Alternativamente, podrías devolver un mailable desde el método toMail
:
use App\Mail\InvoicePaid as InvoicePaidMailable; /** * Get the mail representation of the notification. */ public function toMail(object $notifiable): Mailable { return (new InvoicePaidMailable($this->invoice)) ->to($notifiable->email) ->attachFromStorage('/path/to/file'); }
Cuando sea necesario, se pueden adjuntar múltiples archivos a un mensaje utilizando el método attachMany
:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->greeting('Hello!') ->attachMany([ '/path/to/forge.svg', '/path/to/vapor.svg' => [ 'as' => 'Logo.svg', 'mime' => 'image/svg+xml', ], ]); }
Archivos de Datos en Crudo
El método attachData
puede utilizarse para adjuntar una cadena de bytes en bruto como un archivo adjunto. Al llamar al método attachData
, debes proporcionar el nombre del archivo que se debe asignar al archivo adjunto:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->greeting('Hello!') ->attachData($this->pdf, 'name.pdf', [ 'mime' => 'application/pdf', ]); }
Añadiendo Etiquetas y Metadatos
Algunos proveedores de correo electrónico de terceros como Mailgun y Postmark admiten "etiquetas" y "metadatos" de mensaje, que se pueden usar para agrupar y rastrear correos electrónicos enviados por su aplicación. Puede agregar etiquetas y metadatos a un mensaje de correo electrónico a través de los métodos tag
y metadata
:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->greeting('Comment Upvoted!') ->tag('upvote') ->metadata('comment_id', $this->comment->id); }
Si tu aplicación está utilizando el driver de Mailgun, puedes consultar la documentación de Mailgun para obtener más información sobre etiquetas y metadatos. Del mismo modo, también se puede consultar la documentación de Postmark para obtener más información sobre su soporte para etiquetas y metadatos.
Si tu aplicación está utilizando Amazon SES para enviar correos electrónicos, deberías usar el método metadata
para adjuntar etiquetas "SES" al mensaje.
Personalizando el Mensaje de Symfony
El método withSymfonyMessage
de la clase MailMessage
te permite registrar una función anónima que se invocará con la instancia del mensaje Symfony antes de enviar el mensaje. Esto te da la oportunidad de personalizar profundamente el mensaje antes de que sea entregado:
use Symfony\Component\Mime\Email; /** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->withSymfonyMessage(function (Email $message) { $message->getHeaders()->addTextHeader( 'Custom-Header', 'Header Value' ); }); }
Usando Mailables
Si es necesario, puedes devolver un objeto mailable completo desde el método toMail
de tu notificación. Al devolver un Mailable
en lugar de un MailMessage
, deberás especificar el destinatario del mensaje utilizando el método to
del objeto mailable:
use App\Mail\InvoicePaid as InvoicePaidMailable; use Illuminate\Mail\Mailable; /** * Get the mail representation of the notification. */ public function toMail(object $notifiable): Mailable { return (new InvoicePaidMailable($this->invoice)) ->to($notifiable->email); }
Mailables y Notificaciones Bajo Demanda
Si estás enviando una notificación bajo demanda, la instancia de $notifiable
dada al método toMail
será una instancia de Illuminate\Notifications\AnonymousNotifiable
, que ofrece un método routeNotificationFor
que se puede usar para recuperar la dirección de correo electrónico a la que se debe enviar la notificación bajo demanda:
use App\Mail\InvoicePaid as InvoicePaidMailable; use Illuminate\Notifications\AnonymousNotifiable; use Illuminate\Mail\Mailable; /** * Get the mail representation of the notification. */ public function toMail(object $notifiable): Mailable { $address = $notifiable instanceof AnonymousNotifiable ? $notifiable->routeNotificationFor('mail') : $notifiable->email; return (new InvoicePaidMailable($this->invoice)) ->to($address); }
Previsualizando Notificaciones por Mail
Al diseñar una plantilla de notificación por correo, es conveniente previsualizar rápidamente el mensaje de correo renderizado en tu navegador como una plantilla Blade típica. Por esta razón, Laravel te permite devolver cualquier mensaje de correo generado por una notificación por correo directamente desde una función anónima de ruta o un controlador. Cuando se devuelve un MailMessage
, se renderizará y se mostrará en el navegador, lo que te permitirá previsualizar rápidamente su diseño sin necesidad de enviarlo a una dirección de correo electrónico real:
use App\Models\Invoice; use App\Notifications\InvoicePaid; Route::get('/notification', function () { $invoice = Invoice::find(1); return (new InvoicePaid($invoice)) ->toMail($invoice->user); });
Notificaciones por Mail Markdown
Las notificaciones de correo Markdown te permiten aprovechar las plantillas predefinidas de notificaciones por correo, al mismo tiempo que te brindan más libertad para escribir mensajes personalizados más largos. Dado que los mensajes están escritos en Markdown, Laravel puede renderizar hermosas plantillas HTML responsivas para los mensajes y también generar automáticamente un equivalente en texto plano.
Generando el Mensaje
Para generar una notificación con una plantilla Markdown correspondiente, puedes utilizar la opción --markdown
del comando Artisan make:notification
:
php artisan make:notification InvoicePaid --markdown=mail.invoice.paid
Como todas las demás notificaciones por correo, las notificaciones que utilizan plantillas Markdown deben definir un método toMail
en su clase de notificación. Sin embargo, en lugar de usar los métodos line
y action
para construir la notificación, utiliza el método markdown
para especificar el nombre de la plantilla Markdown que se debe usar. Se puede pasar un array de datos que deseas hacer disponibles para la plantilla como segundo argumento del método:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { $url = url('/invoice/'.$this->invoice->id); return (new MailMessage) ->subject('Invoice Paid') ->markdown('mail.invoice.paid', ['url' => $url]); }
Escribiendo el Mensaje
Las notificaciones por correo en Markdown utilizan una combinación de componentes de Blade y sintaxis Markdown, lo que te permite construir notificaciones fácilmente mientras aprovechas los componentes de notificación predefinidos de Laravel:
<x-mail::message> # Invoice Paid Your invoice has been paid! <x-mail::button :url="$url"> View Invoice </x-mail::button> Thanks,<br> {{ config('app.name') }} </x-mail::message>
Componente de Botón
El componente de botón renderiza un enlace de botón centrado. El componente acepta dos argumentos, una url
y un color
opcional. Los colores admitidos son primary
, green
y red
. Puedes añadir tantos componentes de botón a una notificación como desees:
<x-mail::button :url="$url" color="green"> View Invoice </x-mail::button>
Componente de Panel
El componente del panel renderiza el bloque de texto dado en un panel que tiene un color de fondo ligeramente diferente al del resto de la notificación. Esto te permite llamar la atención sobre un bloque de texto dado:
<x-mail::panel> This is the panel content. </x-mail::panel>
Componente de Tabla
El componente de tabla te permite transformar una tabla Markdown en una tabla HTML. El componente acepta la tabla Markdown como su contenido. Se admite la alineación de columnas de tabla utilizando la sintaxis de alineación de tabla Markdown predeterminada:
<x-mail::table> | Laravel | Table | Example | | ------------- | :-----------: | ------------: | | Col 2 is | Centered | $10 | | Col 3 is | Right-Aligned | $20 | </x-mail::table>
Personalizando los Componentes
Puedes exportar todos los componentes de notificación Markdown a tu propia aplicación para personalizarlos. Para exportar los componentes, utiliza el comando Artisan vendor:publish
para publicar la etiqueta de activo laravel-mail
:
php artisan vendor:publish --tag=laravel-mail
Este comando publicará los componentes de correo Markdown en el directorio resources/views/vendor/mail
. El directorio mail
contendrá un directorio html
y un directorio text
, cada uno conteniendo sus respectivas representaciones de cada componente disponible. Puedes personalizar estos componentes como desees.
Personalizando el CSS
Después de exportar los componentes, el directorio resources/views/vendor/mail/html/themes
contendrá un archivo default.css
. Puedes personalizar el CSS en este archivo y tus estilos se incluirán automáticamente en las representaciones HTML de tus notificaciones Markdown.
Si deseas crear un tema completamente nuevo para los componentes de Markdown de Laravel, puedes colocar un archivo CSS dentro del directorio html/themes
. Después de nombrar y guardar tu archivo CSS, actualiza la opción theme
del archivo de configuración mail
para que coincida con el nombre de tu nuevo tema.
Para personalizar el tema de una notificación individual, puedes llamar al método theme
mientras construyes el mensaje de correo de la notificación. El método theme
acepta el nombre del tema que se debe usar al enviar la notificación:
/** * Get the mail representation of the notification. */ public function toMail(object $notifiable): MailMessage { return (new MailMessage) ->theme('invoice') ->subject('Invoice Paid') ->markdown('mail.invoice.paid', ['url' => $url]); }
Notificaciones de Base de Datos
El canal de notificaciones database
almacena la información de notificación en una tabla de base de datos. Esta tabla contendrá información como el tipo de notificación, así como una estructura de datos JSON que describe la notificación.
Puedes consultar la tabla para mostrar las notificaciones en la interfaz de usuario de tu aplicación. Pero, antes de que puedas hacer eso, necesitarás crear una tabla de base de datos para mantener tus notificaciones. Puedes usar el comando make:notifications-table
para generar una migración con el esquema de tabla apropiado:
php artisan make:notifications-table php artisan migrate
Si tus modelos notifiables están usando claves primarias UUID o ULID, deberías reemplazar el método morphs
con uuidMorphs
o ulidMorphs
en la migración de la tabla de notificaciones.
Formateando Notificaciones de Base de Datos
Si una notificación admite ser almacenada en una tabla de base de datos, debes definir un método toDatabase
o toArray
en la clase de notificación. Este método recibirá una entidad $notifiable
y debe devolver un array PHP plano. El array devuelto será codificado como JSON y almacenado en la columna data
de tu tabla notifications
. Vamos a ver un ejemplo del método toArray
:
/** * Get the array representation of the notification. * * @return array<string, mixed> */ public function toArray(object $notifiable): array { return [ 'invoice_id' => $this->invoice->id, 'amount' => $this->invoice->amount, ]; }
Cuando la notificación se almacena en la base de datos de tu aplicación, la columna type
se rellenará con el nombre de la clase de la notificación. Sin embargo, puedes personalizar este comportamiento definiendo un método databaseType
en tu clase de notificación:
/** * Get the notification's database type. * * @return string */ public function databaseType(object $notifiable): string { return 'invoice-paid'; }
toDatabase
vs. toArray
El método toArray
también es utilizado por el canal broadcast
para determinar qué datos transmitir a tu frontend impulsado por JavaScript. Si deseas tener dos representaciones de array diferentes para los canales database
y broadcast
, debes definir un método toDatabase
en lugar de un método toArray
.
Accediendo a las Notificaciones
Una vez que las notificaciones están almacenadas en la base de datos, necesitas una forma conveniente de acceder a ellas desde tus entidades notificables. El trait Illuminate\Notifications\Notifiable
, que se incluye en el modelo App\Models\User
por defecto de Laravel, incluye una relación notifications
Eloquent relationship que devuelve las notificaciones para la entidad. Para obtener las notificaciones, puedes acceder a este método como cualquier otra relación Eloquent. Por defecto, las notificaciones se ordenarán por la marca de tiempo created_at
, con las notificaciones más recientes al principio de la colección:
$user = App\Models\User::find(1); foreach ($user->notifications as $notification) { echo $notification->type; }
Si deseas recuperar solo las notificaciones "no leídas", puedes usar la relación unreadNotifications
. Nuevamente, estas notificaciones se ordenarán por la marca de tiempo created_at
, con las notificaciones más recientes al principio de la colección:
$user = App\Models\User::find(1); foreach ($user->unreadNotifications as $notification) { echo $notification->type; }
Para acceder a tus notificaciones desde tu cliente JavaScript, debes definir un controlador de notificaciones para tu aplicación que devuelva las notificaciones para una entidad notificable, como el usuario actual. Luego, puedes hacer una solicitud HTTP a la URL de ese controlador desde tu cliente JavaScript.
Marcando Notificaciones como Leídas
Típicamente, querrás marcar una notificación como "leída" cuando un usuario la visualiza. El trait Illuminate\Notifications\Notifiable
proporciona un método markAsRead
, que actualiza la columna read_at
en el registro de la base de datos de la notificación:
$user = App\Models\User::find(1); foreach ($user->unreadNotifications as $notification) { $notification->markAsRead(); }
Sin embargo, en lugar de iterar a través de cada notificación, puedes usar el método markAsRead
directamente en una colección de notificaciones:
$user->unreadNotifications->markAsRead();
También puedes utilizar una consulta de actualización masiva para marcar todas las notificaciones como leídas sin recuperarlas de la base de datos:
$user = App\Models\User::find(1); $user->unreadNotifications()->update(['read_at' => now()]);
Puedes eliminar
las notificaciones para quitarlas de la tabla por completo:
$user->notifications()->delete();
Notificaciones de Difusión
Antes de transmitir notificaciones, debes configurar y conocer los servicios de transmisión de eventos de Laravel. La transmisión de eventos proporciona una forma de reaccionar a eventos de Laravel del lado del servidor desde tu frontend impulsado por JavaScript.
Formateando Notificaciones de Difusión
El canal broadcast
transmite notificaciones utilizando los servicios de broadcasting de eventos de Laravel, permitiendo que tu frontend alimentado por JavaScript reciba notificaciones en tiempo real. Si una notificación admite broadcasting, puedes definir un método toBroadcast
en la clase de notificación. Este método recibirá una entidad $notifiable
y debería devolver una instancia de BroadcastMessage
. Si el método toBroadcast
no existe, se utilizará el método toArray
para reunir los datos que deben ser transmitidos. Los datos devueltos se codificarán como JSON y se transmitirán a tu frontend alimentado por JavaScript. Veamos un ejemplo del método toBroadcast
:
use Illuminate\Notifications\Messages\BroadcastMessage; /** * Get the broadcastable representation of the notification. */ public function toBroadcast(object $notifiable): BroadcastMessage { return new BroadcastMessage([ 'invoice_id' => $this->invoice->id, 'amount' => $this->invoice->amount, ]); }
Configuración de Cola de Transmisión
Todas las notificaciones de transmisión se encolan para su transmisión. Si deseas configurar la conexión de cola o el nombre de cola que se utiliza para encolar la operación de transmisión, puedes usar los métodos onConnection
y onQueue
de BroadcastMessage
:
return (new BroadcastMessage($data)) ->onConnection('sqs') ->onQueue('broadcasts');
Personalizando el Tipo de Notificación
Además de los datos que especifiques, todas las notificaciones de transmisión también tienen un campo type
que contiene el nombre completo de la clase de la notificación. Si deseas personalizar el type
de la notificación, puedes definir un método broadcastType
en la clase de notificación:
/** * Get the type of the notification being broadcast. */ public function broadcastType(): string { return 'broadcast.message'; }
Escuchando Notificaciones
Las notificaciones se transmitirán en un canal privado formateado utilizando una convención .{id}
. Así que, si estás enviando una notificación a una instancia de App\Models\User
con un ID de 1
, la notificación se transmitirá en el canal privado App.Models.User.1
. Al usar Laravel Echo, puedes escuchar fácilmente notificaciones en un canal utilizando el método notification
:
Echo.private('App.Models.User.' + userId) .notification((notification) => { console.log(notification.type); });
Personalizando el Canal de Notificación
Si deseas personalizar en qué canal se transmiten las notificaciones de transmisión de una entidad, puedes definir un método receivesBroadcastNotificationsOn
en la entidad notifiable:
<?php namespace App\Models; use Illuminate\Broadcasting\PrivateChannel; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; class User extends Authenticatable { use Notifiable; /** * The channels the user receives notification broadcasts on. */ public function receivesBroadcastNotificationsOn(): string { return 'users.'.$this->id; } }
Notificaciones SMS
Enviar notificaciones por SMS en Laravel está impulsado por Vonage (anteriormente conocido como Nexmo). Antes de que puedas enviar notificaciones a través de Vonage, necesitas instalar los paquetes laravel/vonage-notification-channel
y guzzlehttp/guzzle
:
composer require laravel/vonage-notification-channel guzzlehttp/guzzle
El paquete incluye un archivo de configuración. Sin embargo, no es necesario que exportes este archivo de configuración a tu propia aplicación. Puedes simplemente usar las variables de entorno VONAGE_KEY
y VONAGE_SECRET
para definir tus claves pública y secreta de Vonage.
Después de definir tus claves, debes configurar una variable de entorno VONAGE_SMS_FROM
que defina el número de teléfono desde el cual se deben enviar tus mensajes SMS por defecto. Puedes generar este número de teléfono dentro del panel de control de Vonage:
VONAGE_SMS_FROM=15556666666
Formateando Notificaciones SMS
Si una notificación admite ser enviada como un SMS, debes definir un método toVonage
en la clase de notificación. Este método recibirá una entidad $notifiable
y debe devolver una instancia de Illuminate\Notifications\Messages\VonageMessage
:
use Illuminate\Notifications\Messages\VonageMessage; /** * Get the Vonage / SMS representation of the notification. */ public function toVonage(object $notifiable): VonageMessage { return (new VonageMessage) ->content('Your SMS message content'); }
Contenido Unicode
Si tu mensaje SMS contendrá caracteres unicode, debes llamar al método unicode
al construir la instancia de VonageMessage
:
use Illuminate\Notifications\Messages\VonageMessage; /** * Get the Vonage / SMS representation of the notification. */ public function toVonage(object $notifiable): VonageMessage { return (new VonageMessage) ->content('Your unicode message') ->unicode(); }
Personalizando el Número "Desde"
Si deseas enviar algunas notificaciones desde un número de teléfono que es diferente del número de teléfono especificado por tu variable de entorno VONAGE_SMS_FROM
, puedes llamar al método from
en una instancia de VonageMessage
:
use Illuminate\Notifications\Messages\VonageMessage; /** * Get the Vonage / SMS representation of the notification. */ public function toVonage(object $notifiable): VonageMessage { return (new VonageMessage) ->content('Your SMS message content') ->from('15554443333'); }
Añadiendo una Referencia de Cliente
Si deseas hacer un seguimiento de costos por usuario, equipo o cliente, puedes añadir una "referencia de cliente" a la notificación. Vonage te permitirá generar informes utilizando esta referencia de cliente para que puedas entender mejor el uso de SMS de un cliente en particular. La referencia de cliente puede ser cualquier cadena de hasta 40 caracteres:
use Illuminate\Notifications\Messages\VonageMessage; /** * Get the Vonage / SMS representation of the notification. */ public function toVonage(object $notifiable): VonageMessage { return (new VonageMessage) ->clientReference((string) $notifiable->id) ->content('Your SMS message content'); }
Enrutando Notificaciones SMS
Para enrutar las notificaciones de Vonage al número de teléfono correcto, define un método routeNotificationForVonage
en tu entidad notifiable:
<?php namespace App\Models; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; use Illuminate\Notifications\Notification; class User extends Authenticatable { use Notifiable; /** * Route notifications for the Vonage channel. */ public function routeNotificationForVonage(Notification $notification): string { return $this->phone_number; } }
Notificaciones de Slack
Requisitos Previos
Antes de enviar notificaciones de Slack, debes instalar el canal de notificaciones de Slack a través de Composer:
composer require laravel/slack-notification-channel
Además, debes crear una Slack App para tu espacio de trabajo de Slack.
Si solo necesitas enviar notificaciones al mismo espacio de trabajo de Slack en el que se creó la aplicación, debes asegurarte de que tu aplicación tenga los scopes chat:write
, chat:write.public
y chat:write.customize
. Si deseas enviar mensajes como tu aplicación de Slack, también debes asegurarte de que tu aplicación tenga el scope chat:write:bot
. Estos scopes se pueden añadir desde la pestaña de gestión de aplicaciones "OAuth & Permissions" dentro de Slack.
A continuación, copia el "Token OAuth de Bot User" de la aplicación y colócalo dentro de un array de configuración slack
en el archivo de configuración services.php
de tu aplicación. Este token se puede encontrar en la pestaña "OAuth & Permissions" dentro de Slack:
'slack' => [ 'notifications' => [ 'bot_user_oauth_token' => env('SLACK_BOT_USER_OAUTH_TOKEN'), 'channel' => env('SLACK_BOT_USER_DEFAULT_CHANNEL'), ], ],
Distribución de Aplicaciones
Si tu aplicación va a enviar notificaciones a espacios de trabajo de Slack externos que son propiedad de los usuarios de tu aplicación, necesitarás "distribuir" tu aplicación a través de Slack. La distribución de la aplicación se puede gestionar desde la pestaña "Gestionar distribución" de tu aplicación dentro de Slack. Una vez que tu aplicación haya sido distribuida, puedes usar Socialite para obtener tokens de bot de Slack en nombre de los usuarios de tu aplicación.
Formateando Notificaciones de Slack
Si una notificación admite ser enviada como un mensaje de Slack, debes definir un método toSlack
en la clase de notificación. Este método recibirá una entidad $notifiable
y debe devolver una instancia de Illuminate\Notifications\Slack\SlackMessage
. Puedes construir notificaciones enriquecidas utilizando la API Block Kit de Slack. El siguiente ejemplo puede ser visualizado en el generador de Block Kit de Slack:
use Illuminate\Notifications\Slack\BlockKit\Blocks\ContextBlock; use Illuminate\Notifications\Slack\BlockKit\Blocks\SectionBlock; use Illuminate\Notifications\Slack\BlockKit\Composites\ConfirmObject; use Illuminate\Notifications\Slack\SlackMessage; /** * Get the Slack representation of the notification. */ public function toSlack(object $notifiable): SlackMessage { return (new SlackMessage) ->text('One of your invoices has been paid!') ->headerBlock('Invoice Paid') ->contextBlock(function (ContextBlock $block) { $block->text('Customer #1234'); }) ->sectionBlock(function (SectionBlock $block) { $block->text('An invoice has been paid.'); $block->field("*Invoice No:*\n1000")->markdown(); $block->field("*Invoice Recipient:*\ntaylor@laravel.com")->markdown(); }) ->dividerBlock() ->sectionBlock(function (SectionBlock $block) { $block->text('Congratulations!'); }); }
Usando la Plantilla del Block Kit Builder de Slack
En lugar de usar los métodos de construcción de mensajes fluentes para construir tu mensaje de Block Kit, puedes proporcionar el payload JSON en bruto generado por el Block Kit Builder de Slack al método usingBlockKitTemplate
:
use Illuminate\Notifications\Slack\SlackMessage; use Illuminate\Support\Str; /** * Get the Slack representation of the notification. */ public function toSlack(object $notifiable): SlackMessage { $template = <<<JSON { "blocks": [ { "type": "header", "text": { "type": "plain_text", "text": "Team Announcement" } }, { "type": "section", "text": { "type": "plain_text", "text": "We are hiring!" } } ] } JSON; return (new SlackMessage) ->usingBlockKitTemplate($template); }
Interactividad de Slack
El sistema de notificaciones Block Kit de Slack ofrece potentes características para manejar la interacción del usuario. Para utilizar estas funciones, tu aplicación de Slack debe tener "Interactividad" habilitada y una "URL de solicitud" configurada que apunte a una URL servida por tu aplicación. Estas configuraciones se pueden gestionar desde la pestaña de gestión de aplicaciones "Interactivity & Shortcuts" dentro de Slack.
En el siguiente ejemplo, que utiliza el método actionsBlock
, Slack enviará una solicitud POST
a tu "URL de Solicitud" con un payload que contiene el usuario de Slack que hizo clic en el botón, el ID del botón clicado y más. Tu aplicación puede luego determinar la acción a tomar basada en el payload. También deberías verificar que la solicitud fue realizada por Slack:
use Illuminate\Notifications\Slack\BlockKit\Blocks\ActionsBlock; use Illuminate\Notifications\Slack\BlockKit\Blocks\ContextBlock; use Illuminate\Notifications\Slack\BlockKit\Blocks\SectionBlock; use Illuminate\Notifications\Slack\SlackMessage; /** * Get the Slack representation of the notification. */ public function toSlack(object $notifiable): SlackMessage { return (new SlackMessage) ->text('One of your invoices has been paid!') ->headerBlock('Invoice Paid') ->contextBlock(function (ContextBlock $block) { $block->text('Customer #1234'); }) ->sectionBlock(function (SectionBlock $block) { $block->text('An invoice has been paid.'); }) ->actionsBlock(function (ActionsBlock $block) { // ID defaults to "button_acknowledge_invoice"... $block->button('Acknowledge Invoice')->primary(); // Manually configure the ID... $block->button('Deny')->danger()->id('deny_invoice'); }); }
Modales de Confirmación
Si deseas que los usuarios confirmen una acción antes de que se realice, puedes invocar el método confirm
al definir tu botón. El método confirm
acepta un mensaje y una función anónima que recibe una instancia de ConfirmObject
:
use Illuminate\Notifications\Slack\BlockKit\Blocks\ActionsBlock; use Illuminate\Notifications\Slack\BlockKit\Blocks\ContextBlock; use Illuminate\Notifications\Slack\BlockKit\Blocks\SectionBlock; use Illuminate\Notifications\Slack\BlockKit\Composites\ConfirmObject; use Illuminate\Notifications\Slack\SlackMessage; /** * Get the Slack representation of the notification. */ public function toSlack(object $notifiable): SlackMessage { return (new SlackMessage) ->text('One of your invoices has been paid!') ->headerBlock('Invoice Paid') ->contextBlock(function (ContextBlock $block) { $block->text('Customer #1234'); }) ->sectionBlock(function (SectionBlock $block) { $block->text('An invoice has been paid.'); }) ->actionsBlock(function (ActionsBlock $block) { $block->button('Acknowledge Invoice') ->primary() ->confirm( 'Acknowledge the payment and send a thank you email?', function (ConfirmObject $dialog) { $dialog->confirm('Yes'); $dialog->deny('No'); } ); }); }
Inspeccionando Bloques de Slack
Si deseas inspeccionar rápidamente los bloques que has estado construyendo, puedes invocar el método dd
en la instancia de SlackMessage
. El método dd
generará y volcará una URL al Block Kit Builder de Slack, que muestra una vista previa de la carga útil y la notificación en tu navegador. Puedes pasar true
al método dd
para volcar la carga útil en bruto:
return (new SlackMessage) ->text('One of your invoices has been paid!') ->headerBlock('Invoice Paid') ->dd();
Enrutando Notificaciones de Slack
Para dirigir las notificaciones de Slack al equipo y canal de Slack apropiados, define un método routeNotificationForSlack
en tu modelo notifiable. Este método puede devolver uno de tres valores:
-
null
- lo que diferencia el enrutamiento al canal configurado en la notificación misma. Puedes usar el métodoto
al construir tuSlackMessage
para configurar el canal dentro de la notificación. - Una cadena que especifica el canal de Slack al que enviar la notificación, por ejemplo,
#support-channel
. - Una instancia de
SlackRoute
, que te permite especificar un token OAuth y un nombre de canal, por ejemplo,SlackRoute::make($this->slack_channel, $this->slack_token)
. Este método debe usarse para enviar notificaciones a espacios de trabajo externos. Por ejemplo, devolver#support-channel
desde el métodorouteNotificationForSlack
enviará la notificación al canal#support-channel
en el espacio de trabajo asociado con el token OAuth del Bot User ubicado en el archivo de configuraciónservices.php
de tu aplicación:
<?php namespace App\Models; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; use Illuminate\Notifications\Notification; class User extends Authenticatable { use Notifiable; /** * Route notifications for the Slack channel. */ public function routeNotificationForSlack(Notification $notification): mixed { return '#support-channel'; } }
Notificando Espacios de Trabajo de Slack Externos
Antes de enviar notificaciones a espacios de trabajo de Slack externos, tu aplicación Slack debe ser distribuida.
Por supuesto, a menudo querrás enviar notificaciones a los espacios de trabajo de Slack pertenecientes a los usuarios de tu aplicación. Para hacerlo, primero necesitarás obtener un token OAuth de Slack para el usuario. Afortunadamente, Laravel Socialite incluye un driver de Slack que te permitirá autenticar fácilmente a los usuarios de tu aplicación con Slack y obtener un token de bot.
Una vez que hayas obtenido el token del bot y lo hayas almacenado en la base de datos de tu aplicación, puedes utilizar el método SlackRoute::make
para enviar una notificación al espacio de trabajo del usuario. Además, tu aplicación probablemente necesitará ofrecer una oportunidad para que el usuario especifique a qué canal se deben enviar las notificaciones:
<?php namespace App\Models; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; use Illuminate\Notifications\Notification; use Illuminate\Notifications\Slack\SlackRoute; class User extends Authenticatable { use Notifiable; /** * Route notifications for the Slack channel. */ public function routeNotificationForSlack(Notification $notification): mixed { return SlackRoute::make($this->slack_channel, $this->slack_token); } }
Localizando Notificaciones
Laravel te permite enviar notificaciones en un locale diferente al locale actual de la solicitud HTTP, e incluso recordará este locale si la notificación está en cola.
Para lograr esto, la clase Illuminate\Notifications\Notification
ofrece un método locale
para establecer el idioma deseado. La aplicación cambiará a este locale cuando se esté evaluando la notificación y luego revertirá al locale anterior cuando la evaluación esté completa:
$user->notify((new InvoicePaid($invoice))->locale('es'));
La localización de múltiples entradas notificables también se puede lograr a través de la fachada Notification
:
Notification::locale('es')->send( $users, new InvoicePaid($invoice) );
Locales Preferidos del Usuario
A veces, las aplicaciones almacenan el locale preferido de cada usuario. Al implementar el contrato HasLocalePreference
en tu modelo notifiable, puedes instruir a Laravel para que use este locale almacenado al enviar una notificación:
use Illuminate\Contracts\Translation\HasLocalePreference; class User extends Model implements HasLocalePreference { /** * Get the user's preferred locale. */ public function preferredLocale(): string { return $this->locale; } }
Una vez que hayas implementado la interfaz, Laravel utilizará automáticamente el locale preferido al enviar notificaciones y mailables al modelo. Por lo tanto, no es necesario llamar al método locale
al usar esta interfaz:
$user->notify(new InvoicePaid($invoice));
Pruebas
Puedes usar el método fake
de la facade Notification
para evitar que se envíen notificaciones. Típicamente, el envío de notificaciones no está relacionado con el código que estás probando. Lo más probable es que sea suficiente simplemente afirmar que Laravel fue instruido para enviar una notificación dada.
Después de llamar al método fake
de la fachada Notification
, puedes afirmar que se ordenó el envío de notificaciones a los usuarios e incluso inspeccionar los datos que recibieron las notificaciones:
<?php use App\Notifications\OrderShipped; use Illuminate\Support\Facades\Notification; test('orders can be shipped', function () { Notification::fake(); // Perform order shipping... // Assert that no notifications were sent... Notification::assertNothingSent(); // Assert a notification was sent to the given users... Notification::assertSentTo( [$user], OrderShipped::class ); // Assert a notification was not sent... Notification::assertNotSentTo( [$user], AnotherNotification::class ); // Assert that a given number of notifications were sent... Notification::assertCount(3); });
<?php namespace Tests\Feature; use App\Notifications\OrderShipped; use Illuminate\Support\Facades\Notification; use Tests\TestCase; class ExampleTest extends TestCase { public function test_orders_can_be_shipped(): void { Notification::fake(); // Perform order shipping... // Assert that no notifications were sent... Notification::assertNothingSent(); // Assert a notification was sent to the given users... Notification::assertSentTo( [$user], OrderShipped::class ); // Assert a notification was not sent... Notification::assertNotSentTo( [$user], AnotherNotification::class ); // Assert that a given number of notifications were sent... Notification::assertCount(3); } }
Puedes pasar una función anónima
a los métodos assertSentTo
o assertNotSentTo
para afirmar que se envió una notificación que pasa una "prueba de verdad" dada. Si al menos una notificación fue enviada que pasa la prueba de verdad dada, entonces la afirmación será exitosa:
Notification::assertSentTo( $user, function (OrderShipped $notification, array $channels) use ($order) { return $notification->order->id === $order->id; } );
Notificaciones a Pedido
Si el código que estás probando envía notificaciones bajo demanda, puedes verificar que la notificación bajo demanda fue enviada a través del método assertSentOnDemand
:
Notification::assertSentOnDemand(OrderShipped::class);
Al pasar una función anónima
como segundo argumento al método assertSentOnDemand
, puedes determinar si se envió una notificación bajo demanda a la dirección de "ruta" correcta:
Notification::assertSentOnDemand( OrderShipped::class, function (OrderShipped $notification, array $channels, object $notifiable) use ($user) { return $notifiable->routes['mail'] === $user->email; } );
Eventos de Notificación
Evento de Envío de Notificación
Cuando se está enviando una notificación, el evento Illuminate\Notifications\Events\NotificationSending
es despachado por el sistema de notificaciones. Esto contiene la entidad "notificable" y la instancia de la notificación misma. Puedes crear escuchadores de eventos para este evento dentro de tu aplicación:
use Illuminate\Notifications\Events\NotificationSending; class CheckNotificationStatus { /** * Handle the given event. */ public function handle(NotificationSending $event): void { // ... } }
La notificación no se enviará si un listener de eventos para el evento NotificationSending
devuelve false
desde su método handle
:
/** * Handle the given event. */ public function handle(NotificationSending $event): bool { return false; }
Dentro de un listener de eventos, puedes acceder a las propiedades notifiable
, notification
y channel
en el evento para aprender más sobre el destinatario de la notificación o la notificación en sí:
/** * Handle the given event. */ public function handle(NotificationSending $event): void { // $event->channel // $event->notifiable // $event->notification }
Evento de Notificación Enviada
Cuando se envía una notificación, el evento Illuminate\Notifications\Events\NotificationSent
event es despachado por el sistema de notificaciones. Esto contiene la entidad "notificable" y la instancia de notificación en sí. Puedes crear escuchas de eventos para este evento dentro de tu aplicación:
use Illuminate\Notifications\Events\NotificationSent; class LogNotification { /** * Handle the given event. */ public function handle(NotificationSent $event): void { // ... } }
Dentro de un listener de eventos, puedes acceder a las propiedades notifiable
, notification
, channel
y response
en el evento para saber más sobre el destinatario de la notificación o la notificación misma:
/** * Handle the given event. */ public function handle(NotificationSent $event): void { // $event->channel // $event->notifiable // $event->notification // $event->response }
Canales Personalizados
Laravel viene con un puñado de canales de notificación, pero es posible que desees escribir tus propios controladores para entregar notificaciones a través de otros canales. Laravel lo hace simple. Para comenzar, define una clase que contenga un método send
. El método debería recibir dos argumentos: un $notifiable
y una $notification
.
Dentro del método send
, puedes llamar a métodos en la notificación para recuperar un objeto de mensaje que sea entendido por tu canal y luego enviar la notificación a la instancia $notifiable
como desees:
<?php namespace App\Notifications; use Illuminate\Notifications\Notification; class VoiceChannel { /** * Send the given notification. */ public function send(object $notifiable, Notification $notification): void { $message = $notification->toVoice($notifiable); // Send notification to the $notifiable instance... } }
Una vez que se haya definido la clase de tu canal de notificación, puedes devolver el nombre de la clase desde el método via
de cualquiera de tus notificaciones. En este ejemplo, el método toVoice
de tu notificación puede devolver cualquier objeto que elijas para representar mensajes de voz. Por ejemplo, podrías definir tu propia clase VoiceMessage
para representar estos mensajes:
<?php namespace App\Notifications; use App\Notifications\Messages\VoiceMessage; use App\Notifications\VoiceChannel; use Illuminate\Bus\Queueable; use Illuminate\Contracts\Queue\ShouldQueue; use Illuminate\Notifications\Notification; class InvoicePaid extends Notification { use Queueable; /** * Get the notification channels. */ public function via(object $notifiable): string { return VoiceChannel::class; } /** * Get the voice representation of the notification. */ public function toVoice(object $notifiable): VoiceMessage { // ... } }