netglue / laminas-mail-utils
A collection of straight-forward validators for Laminas mail messages and some common behaviours for transactional email messages
Installs: 27 643
Dependents: 1
Suggesters: 0
Security: 0
Stars: 0
Watchers: 2
Forks: 0
Open Issues: 0
Requires
- php: >=7.3
- laminas/laminas-mail: ^2.10
- laminas/laminas-mime: ^2.7
- laminas/laminas-validator: ^2.13
Requires (Dev)
- doctrine/coding-standard: ^7.0
- laminas/laminas-component-installer: ^2.1
- laminas/laminas-config-aggregator: ^1.2
- laminas/laminas-servicemanager: ^3.4
- phpunit/phpunit: ^9.0
- roave/security-advisories: dev-master
This package is auto-updated.
Last update: 2024-12-24 23:21:24 UTC
README
Introduction
This is a small package to scratch two primary itches
- Validate
Laminas\Mail\Message
instances according to configurable constraints such as a maximum number of recipients, or a non-empty subject line for example. - Provide simple behaviours and interfaces for email messages that should be sent via popular transactional or marketing email service providers; for example, key/value metadata or tagging/categorisation.
Installation
composer require netglue/laminas-mail-utils
Shipped Validators
Validators can be used as part of your input filter setup, or by creating vendor specific validator chains. Here's an example of a concrete validator chain that would be helpful validating messages to be sent via the Postmark API, which imposes certain restrictions on recipients etc:
<?php declare(strict_types=1); namespace App; use Netglue\Mail\Validator\HasFromAddress; use Netglue\Mail\Validator\HasSubject; use Netglue\Mail\Validator\HasToRecipient; use Netglue\Mail\Validator\TotalFromCount; use Netglue\Mail\Validator\TotalRecipientCount; use Netglue\Mail\Validator\TotalReplyToCount; use Laminas\Validator\ValidatorChain; use Laminas\Validator\ValidatorPluginManager; final class PostmarkMessageValidator extends ValidatorChain { private const MAX_RECIPIENTS = 50; public function __construct(?ValidatorPluginManager $pluginManager = null) { parent::__construct(); if ($pluginManager) { $this->setPluginManager($pluginManager); } $this->configureDefaults(); } private function configureDefaults() : void { $this->attachByName(HasFromAddress::class); $this->attachByName(TotalFromCount::class, ['max' => 1]); $this->attachByName(HasSubject::class); $this->attachByName(HasToRecipient::class); $this->attachByName(TotalRecipientCount::class, ['max' => self::MAX_RECIPIENTS]); $this->attachByName(TotalReplyToCount::class, ['max' => 1]); } }
Shipped Traits & Interfaces
You'll find a collection of interfaces that are pretty minimal but hopefully encapsulate what a lot of transactional email service providers offer when it comes to additional features WRT sending mail - for example, assigning tags or categories to individual messages or turning click tracking or open tracking on and off.
Typically, you make a concrete descendant of Laminas\Mail\Message
and implement whichever interfaces suit the provider best so that you have type safety when working with the capabilities of any given message instance. Again, using Postmark as an example, perhaps something like this:
<?php declare(strict_types=1); namespace App\Postmark\Message; use Netglue\Mail\Message\KeyValueMetadata; use Netglue\Mail\Message\KeyValueMetadataBehaviour; use Netglue\Mail\Message\OpenTracking; use Netglue\Mail\Message\OpenTrackingBehaviour; use Netglue\Mail\Message\TaggableMessage; use Netglue\Mail\Message\TaggableMessageBehaviour; use Laminas\Mail\Message; class MyPostmarkMessage extends Message implements TaggableMessage, KeyValueMetadata, OpenTracking { use OpenTrackingBehaviour; use TaggableMessageBehaviour; use KeyValueMetadataBehaviour { addMetaData as parentAddMetaData; } }
// $message is given to us from some factory or service somewhere if ($message instanceof TaggableMessage) { $someVendorApi->setMessageTag($message->getTag()); }
The plan…
… is to implement vendor specific packages that lever these utilities, with, as you may have guessed, Postmark at the top of the list currently…
The package is currently immature and subject to probable BC breaks and contributions are welcomed if this scratches any itches for you 👍