snicco / better-wp-mail
Keep your sanity when working with mails in WordPress
Installs: 16 472
Dependents: 3
Suggesters: 0
Security: 0
Stars: 7
Watchers: 1
Forks: 0
pkg:composer/snicco/better-wp-mail
Requires
- php: ^7.4|^8.0
- ext-filter: *
- snicco/better-wp-api: ^2.0
Requires (Dev)
- codeception/codeception: ^4.1.29
- lucatume/wp-browser: ~3.1.4
- phpunit/phpunit: ^9.5.13
Conflicts
- snicco/better-wp-cache: <2.0.0-beta.9
- snicco/better-wp-cache-bundle: <2.0.0-beta.9
- snicco/better-wp-cli: <2.0.0-beta.9
- snicco/better-wp-cli-testing: <2.0.0-beta.9
- snicco/better-wp-hooks: <2.0.0-beta.9
- snicco/better-wp-hooks-bundle: <2.0.0-beta.9
- snicco/better-wp-mail-bundle: <2.0.0-beta.9
- snicco/better-wp-mail-testing: <2.0.0-beta.9
- snicco/better-wpdb: <2.0.0-beta.9
- snicco/better-wpdb-bundle: <2.0.0-beta.9
- snicco/blade-bridge: <2.0.0-beta.9
- snicco/blade-bundle: <2.0.0-beta.9
- snicco/content-negotiation-middleware: <2.0.0-beta.9
- snicco/debug-bundle: <2.0.0-beta.9
- snicco/default-headers-middleware: <2.0.0-beta.9
- snicco/eloquent: <2.0.0-beta.9
- snicco/encryption-bundle: <2.0.0-beta.9
- snicco/event-dispatcher: <2.0.0-beta.9
- snicco/event-dispatcher-testing: <2.0.0-beta.9
- snicco/guests-only-middleware: <1.0.0
- snicco/http-routing: <2.0.0-beta.9
- snicco/http-routing-bundle: <2.0.0-beta.9
- snicco/http-routing-testing: <2.0.0-beta.9
- snicco/https-only-middleware: <2.0.0-beta.9
- snicco/illuminate-container-bridge: <2.0.0-beta.9
- snicco/kernel: <2.0.0-beta.9
- snicco/kernel-testing: <2.0.0-beta.9
- snicco/method-override-middleware: <2.0.0-beta.9
- snicco/minimal-logger: <2.0.0-beta.9
- snicco/must-match-route-middleware: <2.0.0-beta.9
- snicco/no-robots-middleware: <2.0.0-beta.9
- snicco/open-redirect-protection-middleware: <2.0.0-beta.9
- snicco/payload-middleware: <2.0.0-beta.9
- snicco/pimple-bridge: <2.0.0-beta.9
- snicco/psr7-error-handler: <2.0.0-beta.9
- snicco/redirect-middleware: <2.0.0-beta.9
- snicco/session: <2.0.0-beta.9
- snicco/session-bundle: <2.0.0-beta.9
- snicco/session-psr16-bridge: <2.0.0-beta.9
- snicco/session-testing: <2.0.0-beta.9
- snicco/session-wp-bridge: <2.0.0-beta.9
- snicco/share-cookies-middleware: <2.0.0-beta.9
- snicco/signed-url: <2.0.0-beta.9
- snicco/signed-url-psr15-bridge: <2.0.0-beta.9
- snicco/signed-url-psr16-bridge: <2.0.0-beta.9
- snicco/signed-url-testing: <2.0.0-beta.9
- snicco/signed-url-wp-bridge: <2.0.0-beta.9
- snicco/str-arr: <2.0.0-beta.9
- snicco/templating: <2.0.0-beta.9
- snicco/templating-bundle: <2.0.0-beta.9
- snicco/testable-clock: <2.0.0-beta.9
- snicco/testing-bundle: <2.0.0-beta.9
- snicco/trailing-slash-middleware: <2.0.0-beta.9
- snicco/wp-auth-only-middleware: <2.0.0-beta.9
- snicco/wp-capability-middleware: <2.0.0-beta.9
- snicco/wp-capapility-middleware: <1.0.0
- snicco/wp-guests-only-middleware: <2.0.0-beta.9
- snicco/wp-nonce-middleware: <2.0.0-beta.9
- dev-master
- v2.0.0-beta.9
- v2.0.0-beta.8
- v2.0.0-beta.7
- v2.0.0-beta.6
- v2.0.0-beta.5
- v2.0.0-beta.4
- v2.0.0-beta.3
- v2.0.0-beta.2
- v2.0.0-beta.1
- v1.10.1
- v1.10.0
- v1.9.1
- v1.9.0
- v1.8.1
- v1.8.0
- v1.7.0
- v1.6.2
- v1.6.1
- v1.6.0
- v1.5.0
- v1.4.2
- v1.4.1
- v1.4.0
- v1.3.0
- v1.2.1
- v1.2.0
- v1.1.3
- v1.1.2
- v1.1.1
- v1.1.0
- v1.0.2
- v1.0.1
- v1.0.0
- dev-beta
This package is auto-updated.
Last update: 2025-10-07 16:56:47 UTC
README
BetterWPMail is a small library that provides an expressive, object-orientated API around
the wp_mail function.
BetterWPMail is not an SMTP-plugin!
It has (optional) support for many mail transports, but will default to using
a WPMailTransport, so that it's usable in distributed WordPress code.
Table of contents
- Motivation
- Installation
- Creating a mailer
- Creating and sending emails
- Testing
- Contributing
- Issues and PR's
- Security
Motivation
To list all problems of the wp_mail function would
take a long time. The most problematic ones are:
- ❌ No support for a plain-text version when sending a html body.
- ❌ No support for inline-attachments.
- ❌ No support for complex multi-part emails.
- ❌ You can't choose a custom filename for attachments.
- ❌ You can't send attachments that you already have in memory (like a generated PDF). You always have to write to a tmp file first.
- ❌ Zero error-handling.
- ❌ No support for templated emails.
- ...
- ...
Many plugins employ massive hacks to circumvent these issues:
This is what you probably find in most WordPress plugin code:
function my_plugin_send_mail(string $to, string $html_message) { add_filter('phpmailer_init', 'add_plain_text'); /* Add ten other filters */ wp_mail($to, $html_message); remove_filter('phpmailer_init', 'add_plain_text') /* Remove ten other filters */ } function add_plain_text(\PHPMailer\PHPMailer\PHPMailer $mailer) { $mailer->AltBody = strip_tags($mailer->Body); }
Why is this so bad?
Besides the fact that you are running a lot of unneeded hooks for every email you sent, what happens if wp_mail
throws an exception that is recovered somewhere else?
You now have ten leftover hook callbacks that modify every outgoing email during the same PHP process. Depending on the kind of filters you added, there is now great potential for bugs that are almost impossible to debug.
A real example of this can be seen here in
the WooCommerce
code base.
(Not bashing WooCommerce here, there is currently no alternative with the way wp_mail works.)
Under the hood WordPress uses the bundled PHPMailer which is a reputable
and rock-solid library.
PHPMailer has native support for most of the problems listed above,
wp_mail
just doesn't use them.
Here is where BetterWPMail comes into play.
Installation
composer require snicco/better-wp-mail
Creating a Mailer
Instead of using wp_mail directly, you'll use
the Mailer class which is able to send Email objects.
Quickstart:
use Snicco\Component\BetterWPMail\Mailer; $mailer = new Mailer();
The full signature of Mailer::__construct is
public function __construct( ?Transport $transport = null, ?MailRenderer $mail_renderer = null, ?MailEvents $event_dispatcher = null, ?MailDefaults $default_config = null )
- 
Transportis an interface and will default to theWPMailTransportwhere all emails will eventually send usingwp_mail.If you are using BetterWPMail in a controlled environment, you can provide your own implementation of the Transportinterface. If you are distributing code you should always use the default transport since you can't control the SMTP-Plugin that your users will have installed.In the future we will create a symfony/mailertransport which will allow you to send emails with any of dozens of providers that Symfony`s mailer integrates with.
- 
The MailRendererinterface is responsible for converting mail templates to html/plain-text content. By default, aFileSystemRendererwill be used, which searches for a file matching the template name.
- 
The MailEventsinterface is responsible for firing events right before and right after an email was sent. By default, an instance ofNullEventswill be used which will not emit any events.
- 
MailDefaultsis responsible for providing fallback configuration for settings sender name, reply-to address etc.
Creating and sending emails
Immutability
The Email class is an immutable value object. You can not change an email once its
created. All public methods on the Email class return a new, modified version of the
object.
Immutability is not common in the PHP community, but it's actually simple to understand:
use Snicco\Component\BetterWPMail\ValueObject\Email; $email = new Email(); ❌ // This is incorrect. $email->addTo('calvin@snicco.io'); ✅ // This is correct $email = $email->addTo('calvin@snicco.io');
The basic convention in BetterWPMail is:
- methods starting with addwill merge attributes and return a new object.
- methods starting with withwill replace attributes and return a new object.
use Snicco\Component\BetterWPMail\ValueObject\Email; $email = new Email(); $email = $email->addTo('calvin@snicco.io'); // The email has one recipient now. $email = $email->addTo('marlon@snicco.io'); // The email has two recipients now. $email = $email->withTo('jondoe@snicco.io'); // The email has one recipient "jondoe@snicco.io"
Sending an email
Emails are sent using the Mailer class.
At minimum, an email needs a recipient and a body (html/text/attachments):
use Snicco\Component\BetterWPMail\ValueObject\Email; $email = (new Email())->addTo('calvin@snicco.io') ->withHtmlBody('<h1>BetterWPMail is awesome</h1>'); $mailer->send($email);
Adding addresses
All the methods that require email addresses (from(), to(), etc.) accept strings, arrays, a WP_User instance or
a MailBox instance
use Snicco\Component\BetterWPMail\ValueObject\Email; use Snicco\Component\BetterWPMail\ValueObject\Mailbox; $email = new Email(); $admin = new WP_User(1); $email = $email // email address is a simple string ->addTo('calvin@snicco.io') // with an explicit display name ->addCc('Marlon <marlon@snicco.io>') // as an array, where the first argument is the email ->addBcc(['Jon Doe', 'jon@snicco.io']) // as an array with a "name" + "email" key ->addFrom(['name' => 'Jane Doe', 'email' => 'jane@snicco.io']) // with an instance of WP_USER ->addFrom($admin) // with an instance of MailBox ->addReplyTo(Mailbox::create('no-reply@snicco.io'));
Setting mail content
You have two options for setting the content of an email:
- By setting it explicitly as a string.
- By setting a template on the email object which will be rendered to html/plain-text before sending.
use Snicco\Component\BetterWPMail\ValueObject\Email; $email = (new Email())->addTo('calvin@snicco.io'); $email = $email ->withHtmlBody('<h1>BetterWPMail is awesome</h1>') ->withTextBody('BetterWPMail supports plain text.') $templated_email = $email ->withHtmlTemplate('/path/to/template-html.php') ->withTextBody('/path/to/template-plain.txt')
If an email has html-content but no explicit text-content, then the html-content will be passed
through strip_tags and be used as the plain-text version.
Adding context to templates
Assuming that we want to send a welcome email to multiple users with the following template:
<?php // path/to/email-templates/welcome.php ?> <h1>Hi <?= esc_html($first_name) ?></h1>, <p>Thanks for signing up to <?= esc_html($site_name) ?></p>
Here we can use the fact that emails are immutable to reuse a base email instance:
use Snicco\Component\BetterWPMail\ValueObject\Email; $email = (new Email()) ->withHtmlTemplate('path/to/email-templates/welcome.php') ->withContext(['site_name' => 'snicco.io']); // Important: don't use withContext here or site_name is gone. $email1 = $email->addContext('first_name', 'Calvin') ->addTo('calvin@snicco.io'); $mailer->send($email1); $email2 = $email->addContext('first_name', 'Marlon'); ->addTo('marlon@snicco.io'); $mailer->send($email2);
This will result in the following two emails being sent:
<h1>Hi Calvin</h1>, <p>Thanks for signing up to snicco.io</p>
<h1>Hi Marlon</h1>, <p>Thanks for signing up to snicco.io</p>
Adding attachments
Attachments can be added to an instance of Email in two ways:
- Attaching a local path on the filesystem.
use Snicco\Component\BetterWPMail\ValueObject\Email; $email = (new Email())->addTo('calvin@snicco.io'); $email = $email ->addAttachment('/path/to/documents/terms-of-use.pdf') // optionally with a custom display name ->addAttachment('/path/to/documents/privacy.pdf', 'Privacy Policy') // optionally with an explicit content-type, ->addAttachment('/path/to/documents/contract.doc', 'Contract', 'application/msword');
- Attaching a binary string or a stream that you already have in memory (a generated PDF for example)
use Snicco\Component\BetterWPMail\ValueObject\Email; $pdf = /* generate pdf */ $email = (new Email())->addTo('calvin@snicco.io'); $email = $email ->addBinaryAttachment($pdf, 'Your PDF', 'application/pdf')
BetterWPMail depends on it's Transport interface to perform the actual sending
of emails. For that reason, no mime-type detection is performed if you don't pass an explicit content-type for an
attachment. This is delegated to the concrete transport implementation.
The WPMailTransport will delegate this task to wp_mail/PHPMailer.
The behaviour of PHPMailer is the following:
- If you pass an explicit mime-type, use that.
- Try to guess the mime-type from the filename.
- If 2. is not possible default to application/octet-streamwhich is defined as "arbitrary binary data".
Embedding Images
If you want to display images inside your email, you must embed them instead of adding them as attachments.
In your email content you can then reference the embedded image with the syntax:
cid: + image embed name
<?php // path/to/email-templates/welcome-with-image.php ?> <h1>Hi <?= esc_html($first_name) ?></h1>, <img src="cid:logo">
use Snicco\Component\BetterWPMail\ValueObject\Email; $email = (new Email()) ->addTo('calvin@snicco.io') ->addContext('first_name', 'Calvin'); $email1 = $email ->addEmbed('/path/to/images/logo.png', 'logo', 'image/png') ->withHtmlTemplate('path/to/email-templates/welcome-with-image.php'); // or with inline html $email2 = $email ->addEmbed('/path/to/images/logo.png', 'logo', 'image/png') ->withHtmlBody('<img src="cid:logo">');
Adding custom headers
use Snicco\Component\BetterWPMail\ValueObject\Email; $email = (new Email()) ->addTo('calvin@snicco.io') // custom headers are string, string key value pairs. // These are not validated in any form. ->addCustomHeaders(['X-Auto-Response-Suppress'=> 'OOF, DR, RN, NRN, AutoReply'])
Configuring emails globally
The default configuration for all your emails is determined by the MailDefaults
class that you pass into the
Mailer class.
If you don't explicitly pass an instance of MailDefaults
when creating your Mailer, they will be created based on the global WordPress settings.
Remember: You can always overwrite these settings on a per-email basis.
use Snicco\Component\BetterWPMail\Mailer; use Snicco\Component\BetterWPMail\ValueObject\MailDefaults; $from_name = 'My Plugin'; $from_email = 'myplugin@site.com'; $reply_to_name = 'My Plugin Reply-To' $reply_to_email = 'myplugin-reply-to@site.com'; $mail_defaults = new MailDefaults( $from_name, $from_email, $reply_to_name, $reply_to_email ); // Other arguments set to default for brevity. $mailer = new Mailer(null, null, null, $mail_defaults);
Extending the Email class
If you are sending the same email in multiple places, you might want to extend the Email
class to preconfigure shared settings in one place.
Creating your custom emails classes has a lot of synergy with mail events.
An example for a custom welcome email:
use Snicco\Component\BetterWPMail\ValueObject\Email; use Snicco\Component\BetterWPMail\ValueObject\Mailbox; class WelcomeEmail extends Email { // You can configure the protected // priorities of the Email class protected ?int $priority = 5; protected string $text = 'We would like to welcome you to snicco.io'; protected ?string $html_template = '/path/to/templates/welcome.php'; public function __construct(WP_User $user) { // configure dynamic properties in the constructor. $this->subject = sprintf('Welcome to snicco.io %s', $user->display_name); $this->to[] = Mailbox::create($user); $this->context['first_name'] = $user->first_name; } } $user = new WP_User(1); $mailer->send(new WelcomeEmail($user));
Using mail events
When you call Mailer::send two types of events are fired.
Right before passing the Email instance to
the configured Transport the SendingEmail event is fired. This
event contains the current Email as a public property which gives you an opportunity to
change its settings before sending.
Right after an email is sent the EmailWasSent event is fired. This event is mainly
useful for logging purposes.
To use mail events you have to pass an instance of MailEvents
when creating your mailer instance.
By default, BetterWPMail comes with an implementation of this interface that uses the WordPress hook system.
use Snicco\Component\BetterWPMail\Event\MailEventsUsingWPHooks; use Snicco\Component\BetterWPMail\Event\SendingEmail; use Snicco\Component\BetterWPMail\Mailer; use Snicco\Component\BetterWPMail\Transport\WPMailTransport; $mailer = new Mailer( null, null, new MailEventsUsingWPHooks() ); add_filter(Email::class, function (SendingEmail $event) { // This will add 'admin@site.com' to every email that is being sent. $event->email = $event->email->addBcc('admin@site.com'); }); add_filter(WelcomeEmail::class, function (SendingEmail $event) { // This will add 'welcome@site.com' to every welcome email that is sent. $event->email = $event->email->addBcc('welcome@site.com'); });
A common use-case of mail events is allowing users to customize specific mails:
// In your code $user = new WP_User(1); $mailer->send(new MyPluginWelcomeMail($user)); // Third-party code: add_filter(MyPluginWelcomeMail::class, function (SendingEmail $event) { // This will overwrite your default template for the "MyPluginWelcomeEmail" only $event->email = $event->email->withHtmlTemplate('path/to/custom/welcome.php'); });
Writing emails in markdown / Using a custom MailRenderer
If you pass no arguments when creating your mailer instance the default renderer will be used which is a combination of:
- The AggregateRenderer(which delegates the rendering to between multipleMailRendererinstances)
- The FilesystemRenderer(which looks for a file that matches the template name set on theEmail)
Let's now create a custom setup:
- We want to render markdown emails and
- Use the FilesystemRendereras a fallback.
First we need a way to convert markdown to HTML.
We will use erusev/parsedown for this task.
composer require erusev/parsedown
Now let's create a custom MarkdownMailRenderer:
use Snicco\Component\BetterWPMail\Renderer\MailRenderer; class MarkdownEmailRenderer implements MailRenderer { // This renderer should only render .md files that exist. public function supports(string $template_name,?string $extension = null) : bool{ return 'md' === $extension && is_file($template_name); } public function render(string $template_name,array $context = []) : string{ // First, we get the string contents of the template. $contents = file_get_contents($template_name); // To allow basic templating, replace placeholders inside {{ }} foreach ($context as $name => $value ) { $contents = str_replace('{{'.$name'.}}', $value); } // Convert the markdown to HTML and return it. return (new Parsedown())->text($contents); } }
Now that we are ready to render markdown emails we can create our Mailer like this:
use Snicco\Component\BetterWPMail\Mailer; use Snicco\Component\BetterWPMail\Renderer\AggregateRenderer; use Snicco\Component\BetterWPMail\Renderer\FilesystemRenderer; use Snicco\Component\BetterWPMail\ValueObject\Email; // This mail renderer will use our new markdown renderer (if possible) and default the filesystem renderer. $mail_renderer = new AggregateRenderer( new MarkdownMailRenderer(), new FilesystemRenderer(), ); $mailer = new Mailer(null, $mail_renderer); $email = new Email(); $email = $email->addTo('calvin@snicco.io'); // This email will be renderer with the default renderer $email_html = $email->withHtmlTemplate('/path/to/templates/welcome.php'); $mailer->send($email_html); // This email will be renderer with our new markdown renderer. $email_markdown= $email->withHtmlTemplate('/path/to/templates/markdown/welcome.md'); $mailer->send($email_markdown);
Handling exceptions
In contrast to wp_mail, calling Mailer::send() will
throw a CantSendEmail exception on failure.
use Snicco\Component\BetterWPMail\Exception\CantSendEmail; use Snicco\Component\BetterWPMail\ValueObject\Email; $email = (new Email())->addTo('calvin@snicco.io') ->withHtmlBody('<h1>BetterWPMail has awesome error handling</h1>'); try { $mailer->send($email); } catch (CantSendEmail $e) { // You can catch this exception if you like, // or let it bubble up depending on your use case. error_log($e->getDebugData()); }
This has numerous advantages over the native way of interacting
with wp_mail:
function handleMailError(WP_Error $error) { // what now? } add_action('wp_mail_failed', 'handleMailError'); $success = wp_mail('calvin@snicco.io', 'wp_mail has bad error_handling'); remove_action('wp_mail_failed', 'handleMailError'); if($success === false) { // what now? }
Testing
BetterWPMail comes with a dedicated testing package that provides a FakeTransport class that you should use during
testing.
First, install the package as a composer dev-dependency:
composer install --dev snicco/better-wp-mail-testing
How you wire the FakeTransport into your Mailer instance during testing greatly depends on how
your overall codebase is set up. You probably want to do this inside your dependency-injection container.
The FakeTranport has the following phpunit assertion methods:
use Snicco\Component\BetterWPMail\Mailer; use Snicco\Component\BetterWPMail\Testing\FakeTransport; use Snicco\Component\BetterWPMail\ValueObject\Email; $mailer = new Mailer($transport = new FakeTransport()); // This fill pass $transport->assertNotSent(WelcomeEmail::class); $mailer->send(new MyPluginWelcomeEmail()); // This will fail now. $transport->assertNotSent(MyPluginWelcomeEmail::class); // This will pass $transport->assertSent(MyPluginWelcomeEmail::class); // This will fail $transport->assertSent(PurchaseEmail::class); // This will fail $transport->assertSentTimes(MyPluginWelcomeEmail:class, 2); $mailer->send(new MyPluginWelcomeEmail()); // This will now pass. $transport->assertSentTimes(MyPluginWelcomeEmail:class, 2); $email = (new Email())->addTo('calvin@snicco.io'); $mailer->send($email); // This will pass $transport->assertSentTo('calvin@snicco.io'); // This will pass $transport->assertNotSentTo('marlon@snicco.io'); $email = (new Email())->addTo('marlon@snicco.io'); $mailer->send($email); // This will now fail. $transport->assertNotSentTo('marlon@snicco.io'); // Using an assertion closure. This will pass. $transport->assertSent(Email::class, function (Email $email) { return $email->to()->has('calvin@snicco.io') });
Intercepting WordPress emails
In addition to faking emails send by your own code that uses the Mailer class, the FakeTransport
also lets you fake all other emails that are sent directly by
using wp_mail.
use Snicco\Component\BetterWPMail\Testing\FakeTransport; use Snicco\Component\BetterWPMail\Testing\WPMail; $transport = new FakeTransport() $transport->interceptWordPressEmails(); // This will pass $transport->assertNotSent(WPMail::class); // No emails will be sent here. wp_mail('calvin@snicco.io', 'Hi calvin', 'Testing WordPress emails was never this easy...'); // This will now fail. $transport->assertNotSent(WPMail::class); // This will pass $transport->assertSent(WPMail::class); // This will pass $transport->assertSent(WPMail::class, function (WPMail $mail) { return 'Hi calvin' === $mail->subject(); }); // This will fail $transport->assertSent(WPMail::class, function (WPMail $mail) { return 'Hi marlon' === $mail->subject(); });
Contributing
This repository is a read-only split of the development repo of the Snicco project.
This is how you can contribute.
Reporting issues and sending pull requests
Please report issues in the Snicco monorepo.
Security
If you discover a security vulnerability within BetterWPMail, please follow our disclosure procedure.