martinpetricko / laravel-database-mail
Laravel package that allows sending emails stored in the database based on fired events.
Fund package maintenance!
MartinPetricko
filamentphp.com/plugins/martin-petricko-database-mail
Requires
- php: ^8.2
- nesbot/carbon: ^3.8
- spatie/laravel-package-tools: ^1.19
Requires (Dev)
- larastan/larastan: ^3.1
- laravel/pint: ^1.21
- nunomaduro/collision: ^8.6
- orchestra/testbench: ^10.0
- pestphp/pest: ^3.7
- pestphp/pest-plugin-arch: ^3.0
- pestphp/pest-plugin-laravel: ^3.1
README
Laravel Database Mail lets you store email templates in your database, link them to events, and automatically send them when those events are dispatched.
For implementation of this package check out FilamentPHP implementation.
Support me
You can support me by buying FilamentPHP implementation of this package.
Installation
You can install the package via composer:
composer require martinpetricko/laravel-database-mail
Publish and run the migrations with:
php artisan vendor:publish --tag="database-mail-migrations"
php artisan migrate
Publish the config file with:
php artisan vendor:publish --tag="database-mail-config"
These are the contents of the published config file:
return [ /** * Register event listener for all TriggersDatabaseMail events, * that sends mails associated with the event. */ 'register_event_listener' => true, /** * Period of time when mail exceptions are pruned. */ 'prune_exceptions_period' => now()->subMonth(), /** * Models that are used by Laravel Database Mail. */ 'models' => [ 'mail_exception' => \MartinPetricko\LaravelDatabaseMail\Models\MailException::class, 'mail_template' => \MartinPetricko\LaravelDatabaseMail\Models\MailTemplate::class, ], /** * Mailable that is used to send the mail from database. */ 'event_mail' => \MartinPetricko\LaravelDatabaseMail\Mail\EventMail::class, /** * Resolvers are used to automatically resolve properties of the event. * These property definitions can be later shown to user as available * variables that can be used in the mail template. */ 'resolvers' => [ \MartinPetricko\LaravelDatabaseMail\Properties\Resolvers\EloquentResolver::class, \MartinPetricko\LaravelDatabaseMail\Properties\Resolvers\BooleanResolver::class, \MartinPetricko\LaravelDatabaseMail\Properties\Resolvers\StringResolver::class, ], /** * Register events that implement TriggersDatabaseMail interface. * Events will be used to trigger the mail and this list * of events can be shown to user as available events. */ 'events' => [ // \Illuminate\Auth\Events\Registered::class, ], ];
Register exceptions reporting in bootstrap/app.php
:
use Illuminate\Foundation\Configuration\Exceptions; use MartinPetricko\LaravelDatabaseMail\Exceptions\DatabaseMailException; use MartinPetricko\LaravelDatabaseMail\Facades\LaravelDatabaseMail; //... ->withExceptions(function (Exceptions $exceptions) { $exceptions->report(function (DatabaseMailException $e) { LaravelDatabaseMail::logException($e); }); })
Enable exceptions table pruning:
Schedule::command('model:prune', [ '--model' => [ MartinPetricko\LaravelDatabaseMail\Models\MailException::class, ], ])->daily();
Usage
Create Events
Add TriggersDatabaseMail
interface and CanTriggerDatabaseMail
trait to your standard laravel events.
namespace App\Events; use App\Models\User; use Illuminate\Foundation\Events\Dispatchable; use Illuminate\Queue\SerializesModels; use MartinPetricko\LaravelDatabaseMail\Attachments\Attachment; use MartinPetricko\LaravelDatabaseMail\Events\Concerns\CanTriggerDatabaseMail; use MartinPetricko\LaravelDatabaseMail\Events\Contracts\TriggersDatabaseMail; use MartinPetricko\LaravelDatabaseMail\Recipients\Recipient; class Registered implements TriggersDatabaseMail { use Dispatchable; use SerializesModels; use CanTriggerDatabaseMail; /** * All public properties of the event will be passed * to the mail template body and subject. */ public function __construct(public User $user, public string $emailVerificationUrl) { // } /** * Name of the event that can be used in the UI. */ public static function getName(): string { return 'User Registered'; } /** * Description of the event that can be used in the UI. */ public static function getDescription(): ?string { return 'Fires when a user is registered'; } /** * List of possible recipients that can receive the email. * MailTemplate stores recipient keys that will * receive the email when event is triggered. * * @return Recipient<Registered>[] */ public static function getRecipients(): array { return [ 'user' => new Recipient('Registered User', fn (Registered $event) => [ $event->user, ]), ]; } /** * List of possible attachments that can be attached to the email. * MailTemplate stores attachment keys that will be attached * to the email when the event is triggered. * * @return Attachment<Registered>[] */ public static function getAttachments(): array { return [ 'terms-of-service' => new Attachment('Terms of Services', fn (Registered $event) => [ \Illuminate\Mail\Attachment::fromUrl('https://my-project.com/tos')->as('tos.pdf'), ]), ]; } }
Register Events
Add a list of events to your published config/database-mail.php
file:
'events' => [ \App\Events\Registered::class, ],
Create Mail Template
use \MartinPetricko\LaravelDatabaseMail\Models\MailTemplate; $mailTemplate = MailTemplate::make(); // Internal name of the mail $mailTemplate->name = 'Verify Email Address'; // The event that triggers the mail $mailTemplate->event = \App\Events\Registered::class; // The subject of the email, rendered with blade $mailTemplate->subject = 'Welcome {{ $user->name }}'; // The body of the email, rendered with blade $mailTemplate->body = <<<HTML <h1>Welcome {{ \$user->name }}</h1> <p>Please verify your email address by clicking <a href="{{ \$emailVerificationUrl }}">Here</a></p> HTML; // Keys of the recipients, defined in the event, that will receive the email $mailTemplate->recipients = ['user']; // Keys of the attachments, defined in the event, that will be attached to the email $mailTemplate->attachments = ['terms-of-service']; // Optionally, you can set a delay for how long the mail should be sent after the event is fired $mailTemplate->delay = '1 day 5 hours'; // Determines if mail is sent when event is fired $mailTemplate->is_active = true; $mailTemplate->save();
Dispatch Event
Dispatch the event as you would any other Laravel event with its parameters.
use App\Events\Registered; // ... your bussiness logic Registered::dispatch($registeredUser, $registeredUserEmailVerificationUrl);
List Events
You can get all events that are registered in config/database-mail.php
.
use MartinPetricko\LaravelDatabaseMail\Facades\LaravelDatabaseMail; LaravelDatabaseMail::getEvents();
You can get array of property definitions that can be shown to user as available variables that can be used in mail template.
use MartinPetricko\LaravelDatabaseMail\Facades\LaravelDatabaseMail; LaravelDatabaseMail::getEventAttributes(\App\Events\Registered::class);
Import/Export Mail Templates
You can prepare your mail templates before deploying your application to production and then import them in your seeders.
Export Mail Templates
php artisan mail:export
Import Mail Templates
php artisan mail:import
Seeder Setup
use Illuminate\Support\Facades\Artisan; public function run(): void { /** * Import all mail templates from json files and replace localhost url with production url. */ Artisan::call('mail:import', [ '--all' => true, '--search' => 'http:\/\/localhost', '--replace' => config('app.url'), ]); }
Testing
composer test
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
Please see CONTRIBUTING for details.
Security Vulnerabilities
Please review our security policy on how to report security vulnerabilities.
Credits
License
The MIT License (MIT). Please see License File for more information.