railroad / mailora
Wrapper for Laravel's email functionality that adds HTTP API and front-end-dev-friendly view-creation and use.
Installs: 79 120
Dependents: 1
Suggesters: 0
Security: 0
Stars: 0
Watchers: 1
Forks: 0
Open Issues: 0
pkg:composer/railroad/mailora
Requires
- php: ^8.2
- doctrine/dbal: ^3.0
- guzzlehttp/guzzle: ^7.2
- laravel/framework: ^11.9
Requires (Dev)
- orchestra/testbench: ^9.2
- phpunit/php-code-coverage: ^11.0.1
- phpunit/phpunit: ^11.0.1
README
Wrapper for Laravel's email functionality that adds HTTP API and front-end-dev-friendly view-creation and use.
Table of Contents:
- Mailora
- 1 - Installation and Configuration
- 2 - Features
- 3 - API Reference
 
1 - Installation and Configuration
1.1 - Installation
Install using Composer by running
$ composer require railroad/mailora
or add to composer.json
{
    "require": {
        "railroad/mailora": "dev-master"
    }
}
and then run composer update railroad/mailora.
(I'm not sure what to do regarding specifying which version to use. Maybe just use "dev-master"?)
Add \Railroad\Mailora\Providers\MailoraServiceProvider::class, to the providers array in your application's config/app.php.*
run php artisan vendor:publish, selecting this package from the resultant list of options.
This will register views and routes, and will copy the configuration file into your application. Commit the config file to you application. Detailes on configuration are below.
* (minor note RE future dev of this project) We shouldn't need to do this, but it seems feature of Laravel where the Service provider can just be registered in a package's own composer.json file... is not working, and so therefore you must add this to the application.
1.2 - Configuration
There are two files to configure. They are both in you application's "config" directory:
- 'mail.php'
- 'mailora.php'
See more details about each in the two sections below.
Note that both config files can use Laravel's "env()" helper function. Thus, you can override a value hardcoded in a config file at any time by supplying an environment variable. This can be useful for alternate configurations for local or staging environments.
You can then call these configuration values using Laravel's config helper
Example:
// retrieve a value directly using dot notation $senderAddress = config('mail.defaults.sender-address'); // or retrieve an array and access items as needed. $mail = config('mail.defaults'); $senderName = $mail['sender-name'];
1.2.1 - Laravel-native 'config/mail.php'
This is a Laravel-native file to provide email-sending-service details. This file configures standard Laravel functionality and you can refer to their documentation for details.
Supply the secret for your chosen email service as an environment variable, do not commit the actual value to a config file.
You can hard-code other non-sensitive values in the config file.
1.2.2 - Package provided 'config/mailora.php'
1.2.2.1 - TL;DR
The bare minimum to provide to config/mailora.php is the following:
- 'safety-recipient'
- 'defaults'
- 'sender-address'
- 'recipient-address'
 
- one or both of the following
- 'auth_middleware'
- one or both of the following
- 'approved-recipients'
- 'approved-recipient-domains'
 
 
1.2.2.2 - Details
This file is copied from this package to your application's "config" directory by running artisan vendor:publish.
There are two basic functions. One is sending by calling an unprotected route. This is available to the whole world, and so will only send to email addresses (or domains) explicitly allowed by configuration. The other ("secure") is only accessible to authenicated users. For the former you must supply 'approved-recipients' or 'approved-recipient-domains' values and use the public route. For the latter you must supply a authentication middleware to use the "secure" route.
1.2.2.2.1 - Top-level values
| key | requires app-specific config values | notes regarding requirement | description | 
|---|---|---|---|
| safety-recipient | yes | nothing will work without this | email to send to when environment is not production | 
| approved-recipients | yes* | required for public-route functionality | list (array) of email addresses that emails can be sent to when publicly-available route is used | 
| auth_middleware | yes* | required for auth-protected-route functionality | names of your applications authentication middleware behind which you wish to guard the totally open not publicly-accessible route. See next section for more detailed description | 
| views-root-directory | no* | no change required unless view files created use non-standard path | path of your application's viewsdirectory. From application root (as returned by Laravel'sbase_path()helper). Defaults to '/resources/views/' | 
| views-email-directory | no* | no change required unless view files created use non-standard path | path of directory with your applications custom email templates. *relative to views directory (this is the value of the "views-root-directory" config detailed above). Defaults to 'emails' | 
| mailables-namespace | no* | no change required unless classes created use non-standard namespace | namespace of Mailable classes in your application. | 
| name-of-production-env | no* | no change required unless your production environment is not called "production" | if application's "environment" is anything other than this value then emails will only be sent to the address provided by the "safety-recipient" config value | 
| public-free-for-all | truewould allow public-route to have "send-to" address specified in request. Creates a publicly accessible free-email-sending service that's ripe for exploitation should anybody discover it. You probably shouldn't do this. | ||
| admin | email address to send errors messages to | 
1.2.2.2.2 - values nested under "defaults"
| key | requires app-specific config values | notes regarding requirement | description | 
|---|---|---|---|
| sender-address | yes | nothing will work without this | if no sender-address specified in request use this one | 
| sender-name | if no sender-name and no sender-address specified in request use this name. Will not apply to requests where sender-address supplied. | ||
| recipient-address | yes | nothing will work without this | if no recipient-address specified in request use this one | 
| subject | if no subject specified in request use this one. If this not configured package hard-coded default used. | ||
| success-message | if no recipient-address specified in request use this one If this not configured package hard-coded default used. | ||
| error-message | if no error-message specified in request use this one If this not configured package hard-coded default used. | ||
| type | if no type specified in request use this one If this not configured package hard-coded default used. | ||
| users-email-set-reply-to | if authentication-protected route used and no reply-to address provided in request user's email address is set as reply-to | 
* with caveat—see note
1.2.3 - authentication middleware
Supply one value in a string as "auth_middleware". Use the key from the "$routeMiddleware" property of your application's App\Http\Kernel class (configured as per Laravel functionality).
For example, if your app/Http/Kernel.php has:
    protected $routeMiddleware = [
        'auth' => \WordpressAuthMiddleware::class,
        'auth-special' => \WordpressSpecialAuthMiddleware::class,
        'requires.edge' => \App\Http\Middleware\RequiresEdge::class,
        'requires.edge-or-pack' => \App\Http\Middleware\RequiresPackOrEdge::class,
        'auth.admin' => \WordpressAdminAuthMiddleware::class,
        'auth.executives' => \StatisticsWhiteListFilter::class,
        'auth.shippers' => \App\Http\Middleware\ShippersOrAdmin::class,
    ];
...then in config/mailora.php you can have something like this:
return [ // ... 'auth_middleware' => ['is-logged-in-or-something'], // ... ];
1.3 - Config for local-development
Rather than mess around with changes to the config file, because Laravel's env() function is used, you can just provide environment variable to override anything hard-coded in the config files. Define values in a ".env" file in your application root.
At a bare minimum, you provide "MAILORA_SAFETY_RECIPIENT" so that you're sure of where emails are being sent. Emails will only be sent to that address when the enviroment is is anything other than "production" (or the value returned by config('mailora.name-of-production-env') if it's different than "production");
example:
MAILORA_SAFETY_RECIPIENT=jonathan+mailora_dev_SAFETY_RECIPIENT@drumeo.com
Or supply many values:
MAILORA_SAFETY_RECIPIENT=joe+foo@foo.com
MAILORA_APPROVED_FROM_PUBLIC_RECIPIENTS=['joe+test89347832@foo.com']
MAILORA_NAME_OF_PROD_ENV='staging'
MAILORA_PUBLIC_FREE_FOR_ALL=true
MAILORA_DEFAULT_ADMIN=joe+bar@foo.com
MAIL_FROM_ADDRESS=joe+baz@foo.com
MAIL_FROM_NAME="Joe Black"
MAILORA_DEFAULT_RECIPIENT=joe+qux@foo.com
MAILORA_DEFAULT_TYPE='foo'
2 - Features
2.1 - send email with POST requests to endpoint
To send an email, simply send a POST request as described in these docs. You'll receive a simple "sent" boolean value in response.
2.2 - configure default values for common operations
Always send to the same email address. Set that as the default in the configuration, and then on requests to send to that address, don't pass a "recipient-address" value. The default will be used.
In fact, the endpoint has no required parameters—you can place all required information in the configuration and only provide unique info when required.
2.3 - "all-you-can-eat view variables"
You can easily pass values you need and have them available in your custom view. No PHP needed.
2.3.1 - Accessing Values Passed to View
Values passed to view in request are nested in $input variable.
Thus, if you pass:
$.ajax({ url: 'https://www.foo.com/mailora/secure/send' , type: 'get', dataType: 'json', data: { 'type': 'layouts/action', 'lines': [ 'foo_lines_bar_line_1', 'foo_lines_bar_line_2', 'foo_lines_bar_line_3', 'foo_lines_bar_line_4', 'foo_lines_bar_line_5', 'foo_lines_bar_line_6' ], 'callToAction': { 'url': 'foo_callToAction_bar', 'text': 'foo_callToAction_bar', }, 'logo': 'foo_logo_bar', 'brand': 'foo_brand_bar', 'subject': 'important action!!' }, success: function(response) { /* handle error */ }, error: function(response) { /* handle error */ } });
You can access the custom values you're passing via the $input variable in the view. For example:
<table> @foreach($input['lines'] as $line) <tr><td>{{ $line }}</td></tr> @endforeach @if(!empty($input['callToAction'])) <tr><td><a href="{{ $input['callToAction']['url'] }}">{{ $input['callToAction']['text'] }}</a></td></tr> @endif <tr><td>— The {{ $input['brand'] ?? 'Musora' }} Team</td></tr> </table>
Note the $line variable. This is defined in the foreach-loop, and thus is not $input[line].
2.2 - Easy Custom Views
If no 'type' value is provided in a request, the Mailora's General class and general.blade.php view will be used. If a type value is passed, Mailora will look for a class matching the CapitalizedCamelCase version of that value*. Mailora will look for this class in the namespace returned by config('mailora.mailables-namespace)**. If a class exists there, it will be used. If no class is used, Mailora's General class will be used. This class is simply uses whatever view supplied, and passes a $input parameter along to the view so that all values are retrieved from that.
This enables the following.
Regardless of what class is used—that is to say even if no Mailable class was found matching the ConvertedCapitalizedCamelCase 'type' value passed, Mailora will then look for a view file matching that type value. It looks for files in the directory described by the string returned by config('mailora.views-directory')**. If one if found, that is used. If not, mailora's general.php is used***.
The upshot of all this is that new email-templates can be created with no back-end modifications required. Simply create a view file, and supply and retrieve values from the $input parameter.
TL;DR Regarding the value 'type' specified in request data. It will resolve to class if possible. Else, the default General PHP Mailable class will be used, and 'type' will resolve to a view, if possible. Else, the default 'general.blade.php' will be used.
* the value foo-bar-baz would use the class FooBarBaz and the view file "foo-bar-baz.blade.php".
** this is set—and can therefore be changed—in the mailora.php config file
*** Note that this file exists in "/your-application/vendor/railroad/mailora/resources/views/"
2.2.1 - Flowchart for class and view to use
is type defined in request? → no → use 'general' view and 'General' Mailable class
↓
yes
↓
is type defined in request as 'general'? → yes → use 'general' view and 'General' Mailable class
↓
no
↓
is there a Mailable class that matches a CapitalizedCamelCase version of the 'type' value supplied? → yes → use that
↓
no
↓
use 'General'
↓
is there a view file matching the 'type' value passed? → no → use mailora's general.blade.php
↓
yes
↓
use that
2.2.2 - View files nested in email-directory sub-directories
To specify a view in a subdirectory (of your emails directory), simply type that relative path of the view file.
For Example, let's say your views dir is: /appFoo/resources/views/emails. You can use the view file /appFoo/resources/views/emails/fooView.blade.php by specifying the "type" value for your email request as "fooView". But what if you have a view file in the emails/barDir directory? Say maybe something like /appFoo/resources/views/emails/barDir/quxView.blade? Well, you would just specify the type as barDir/quxView.
This is all of course assuming you don't need any getting or special transformations possible with a custom laravel Mailable PHP class and are okay with the default Mailora General PHP class being used.
3 - API Reference
There are two endpoints:
- POST /mailora/public/send(Publicly accessible)
- POST /mailora/secure/send(User must be authenticated to access this endpoint)
See details below.
In addition to the parameters listed below, any other parameter can be passed, and it will be available in the view!
For example, if you have a 'foo' : 'bar' item in data for a request, in the view, {{ $input['foo'] }} will print "bar".
Another example: 'foo' : 1 in the request will allow {{ $input['foo'] ? 'true' : 'false' }} in the view.
3.1 - Send email from anywhere
POST /mailora/public/send/
Can be called from anywhere. Handy for sending emails from publicly-accessible support and sales pages.
Recipient cannot be specified. Will be sent to MAIL_FROM_ADDRESS unless MAIL_FROM_ADDRESS_PUBLIC provided. Though, you can set the "Sender name"
User must be defined in config file "config/mailora.php". If user is not present there, email will not be send to intended recipient.
3.1.1 - Request Example
let data = { "type": "foo", "sender-address": "bar@some-domain.com", "sender-name": "Baz Qux", "recipient": [ "foo+test_0@bar.com", "foo+test_1@bar.com", { "name":"foo_test_2", "address":"foo+test_2@bar.com" }, { "name":"foo_test_3", "address":"foo+test_3@bar.com" }, { "name":"foo_test_4", "address":"foo+test_4@bar.com" } ], "subject": "Grault garply waldo", "reply-to": "bar@some-domain.com", "users-email-set-reply-to": "", "message": "Fred plugh thud, mos henk. Def." }; $.ajax({ url: 'https://www.foo.com/mailora/secure/send' , type: 'get', dataType: 'json', data: data, success: function(response) { /* handle error */ }, error: function(response) { /* handle error */ } });
Alternatively, set "recipient-address" (and optionally "recipient-name") rather than "recipient" if you just have
one to specify.
let data = { "type": "foo", "sender-address": "bar@some-domain.com", "sender-name": "Baz Qux", "recipient-address": "foo+test_0@bar.com", "subject": "Grault garply waldo", "reply-to": "bar@some-domain.com", "users-email-set-reply-to": "", "message": "Fred plugh thud, mos henk. Def." };
3.1.2 - Request Parameters
Provide all parameters in the request body.
Note that no fields are required!!
| key | required | default can be defined in config('mailora.default') | default hardcoded in package so needn't be provided by config | description|notes | |
|---|---|---|---|---|---|
| type | no | yes | 'general' | ||
| sender-address | no | yes | Email will not be sent if this not provided from request or config. | ||
| sender-name | no | yes | See "Note 1" below | ||
| recipient-address | no | yes | Email will not be sent if this not provided from request or config. | ||
| recipient-name | no | yes | See "Note 1" below | ||
| recipient | no | no (not yet) | See "Note 2" below | ||
| subject | no | yes | 'General Inquiry - Subject not specified' | ||
| reply-to | no | ||||
| users-email-set-reply-to | no | yes | false | If no reply-to param supplied in request but a logged in user is available the 'reply-to' will be set with that user's email address | |
| message | no | yes | ''(empty string) | 
Note 1: A provided name is not used unless address also provided from same source. For example: Say sender-address and sender-name are both set in the configuration file. If a request doesn't not specify request-address, then the addresss and name from the config will be used. However, if the request supplies an address but no name, then that address (from the request) will be used, but not the name from config. The only time the name in the config is used, is when the address in the config is used. When an address is provided in the request, only a name similarily provided in the request will be used. A name provided from config will not be used if an email address is provided in a request.
Note 2: Pass any number of recipients as items in array formatted as JSON-string. Each recipient is an item in the array. There are two options to specify each recipient in the array:
- as just a string of the address.
- as an object-literal with the "address"defined, and optionally the"name"
Example:
[ "foo+test_0@bar.com", "foo+test_1@bar.com", { "name":"foo_test_2", "address":"foo+test_2@bar.com" }, { "name":"foo_test_3", "address":"foo+test_3@bar.com" }, { "name":"foo_test_4", "address":"foo+test_4@bar.com" } ]
Also, this will be renamed to "recipients" one day.
3.1.3 - Response Example
3.1.3.1 -  {200 OK}
{"sent":true}
3.1.3.2 - {500 Internal Server Error}
{"sent":false}
or
{"error":true}
fin
glhf