如何使用 Forever 搭配 Laravel Queue 寄送 Email?

   2016-12-01 0
核心提示:傳統寄送 email 是採用同步的方式,也就是當你寄出一封信,必須等 email server 回應後,才可以繼續後續的程式動作,因此使用者會有明顯的等待時間;若能搭配 queue 機制,寄送 email 後,馬上以非同步的方式回到原來程式繼續執行,會有另外一個 process 去消

傳統寄送 email 是採用同步的方式,也就是當你寄出一封信,必須等 email server 回應後,才可以繼續後續的程式動作,因此使用者會有明顯的等待時間;若能搭配 queue 機制,寄送 email 後,馬上以非同步的方式回到原來程式繼續執行,會有另外一個 process 去消耗 queue,負責寄送 email。

Motivation

Laravel 的 Mail::send() 是以同步方式寄送 email,會有明顯的等待時間;而 Mail:queue() 則是以非同步的方式寄送 email,速度非常快,不過必須搭配 queue 以及其他配套機制。

Version

PHP 7.0.8

Laravel 5.2.45

實際案例

先以 Mail::send() 透過 Gmail 寄送信件,然後再改用 Mail::queue() 方式寄送信件。

單元測試 : 由 Sync 寄送信件

MailServiceTest.php 1 1 GitHub Commit : 單元測試 : 由 sync 寄送信件

tests/MailServiceTest.php
use App\Services\MailService;

class MailServiceTest extends TestCase
{
    /** @test */
    public function 由Sync寄送信件()
    {
        /** arrange */

        /** act */
        App::call(MailService::class . '@mailBySync');

        /** assert */
        $this->assertTrue(true);
    }
}

這裡沒做任何測試,只是透過 PHPUnit 啟動 sync 方式寄送信件。

設定使用 Gmail 寄信

.env 2 2 GitHub Commit : 設定使用 Gmail 寄信

.env
APP_ENV=local
APP_DEBUG=true
APP_KEY=base64:PABSMLELX37jW2jJdwm9Fk6LvUWupulQXAWDZcfA7xE=
APP_URL=http://localhost

DB_CONNECTION=sqlite
DB_HOST=127.0.0.1
DB_PORT=3306
#DB_DATABASE=homestead
DB_USERNAME=homestead
DB_PASSWORD=secret

CACHE_DRIVER=file
SESSION_DRIVER=file
QUEUE_DRIVER=sync

REDIS_HOST=127.0.0.1
REDIS_PASSWORD=null
REDIS_PORT=6379

MAIL_DRIVER=smtp
MAIL_HOST=smtp.gmail.com
MAIL_PORT=587
MAIL_USERNAME=[your gmail address]
MAIL_PASSWORD=[your password]
MAIL_ENCRYPTION=tls

21 行

MAIL_DRIVER=smtp
MAIL_HOST=smtp.gmail.com
MAIL_PORT=587
MAIL_USERNAME=[your gmail address]
MAIL_PASSWORD=[your password]
MAIL_ENCRYPTION=tls
  • MAIL_HOST : 設定為 smtp.gmail.com
  • MAIL_PORT : 設定為 587
  • MAIL_USERNAME : 設定你的 Gmail 帳號,如 example@gmail.com
  • MAIL_PASSWORD : 設定你的 Gmail 密碼。
  • MAIL_ENCRYPTION : 設定為 tls

由 Sync 寄送信件

MailService.php 3 3 GitHub Commit : 由 sync 寄送信件

tests/MailService.php
namespace App\Services;

use Illuminate\Mail\Message;
use Mail;

class MailService
{
    public function mailBySync()
    {
        Mail::send('welcome', [], function (Message $message) {
            $message->sender('oomusou@gmail.com');
            $message->subject('Laravel 5.2 mail by Sync');
            $message->to('oomusou@gmail.com');
        });
    }
}

使用 Mail::send() 寄送信件。

  • 第 1 個參數為 blade 檔案名稱。
  • 第 2 個參數為要傳給 blade 的陣列,若不傳任何資料,也要傳一個空陣列。
  • 第 3 個參數為 closure,主要是 Laravel 希望你將 Message 物件資料填滿。

實際執行會發現, Mail::send() 會需要等 1 到 2 秒才會執行完,因為是同步,要等 smtp server 回應後才會繼續執行。

單元測試 : 由 Queue 寄送信件

MailServiceTest.php 4 4 GitHub Commit : 單元測試 : 由 queue 寄送信件

tests/MailServiceTest.php
use App\Services\MailService;

class MailServiceTest extends TestCase
{
	/** @test */
    public function 由queue寄送信件()
    {
        /** arrange */

        /** act */
        App::call(MailService::class . '@mailByQueue');

        /** assert */
        $this->assertTrue(true);
    }
}

這裡沒做任何測試,只是透過 PHPUnit 啟動 queue 方式寄送信件。

設定本機環境使用 Queue

.env 5 5 GitHub Commit : 設定本機環境使用 queue

.env
APP_ENV=local
APP_DEBUG=true
APP_KEY=base64:PABSMLELX37jW2jJdwm9Fk6LvUWupulQXAWDZcfA7xE=
APP_URL=http://localhost

DB_CONNECTION=sqlite
DB_HOST=127.0.0.1
DB_PORT=3306
#DB_DATABASE=homestead
DB_USERNAME=homestead
DB_PASSWORD=secret

CACHE_DRIVER=file
SESSION_DRIVER=file
#QUEUE_DRIVER=sync
QUEUE_DRIVER=database

REDIS_HOST=127.0.0.1
REDIS_PASSWORD=null
REDIS_PORT=6379

MAIL_DRIVER=smtp
MAIL_HOST=smtp.gmail.com
MAIL_PORT=587
MAIL_USERNAME=[your gmail address]
MAIL_PASSWORD=[your password]
MAIL_ENCRYPTION=tls

15 行

#QUEUE_DRIVER=sync
QUEUE_DRIVER=database

QUEUE_DRIVERsync 改成 database

設定測試環境使用 Queue

phpunit.xml 6 6 GitHub Commit : 設定測試環境使用 queue

phpunit.xml
<?xml version="1.0" encoding="UTF-8"?>
<phpunit backupGlobals="false"
         backupStaticAttributes="false"
         bootstrap="bootstrap/autoload.php"
         colors="true"
         convertErrorsToExceptions="true"
         convertNoticesToExceptions="true"
         convertWarningsToExceptions="true"
         processIsolation="false"
         stopOnFailure="false">
    <testsuites>
        <testsuite name="Application Test Suite">
            <directory suffix="Test.php">./tests</directory>
        </testsuite>
    </testsuites>
    <filter>
        <whitelist processUncoveredFilesFromWhitelist="true">
            <directory suffix=".php">./app</directory>
            <exclude>
                <file>./app/Http/routes.php</file>
            </exclude>
        </whitelist>
    </filter>
    <php>
        <env name="APP_ENV" value="testing"/>
        <env name="QUEUE_DRIVER" value="database"/>
        <env name="DB_CONNECTION" value="sqlite"/>
    </php>
</phpunit>

24 行

<php>
	<env name="APP_ENV" value="testing"/>
    <env name="QUEUE_DRIVER" value="database"/>
    <env name="DB_CONNECTION" value="sqlite"/>
</php>
  • CACHE_DRIVERSESSION_DRIVER 部分刪除。
  • QUEUE_DRIVER 改成 database
  • DB_CONNECTION 改成 sqlite

CACHE_DRIVERSESSION_DRIVER 必須刪除,否則無法再測試環境使用非同步方式寄送 email。

儘管我們之前已經在 .envQUEUE_DRIVER 改成 database ,但 phpunit.xmlQUEUE_DRIVERsync 設定會覆蓋掉 .env 的設定,所以這邊也要改成 database

不能如一般單元測試的 DB_CONNECTION 設定成 sqlite_testing ,也就是將資料庫設定成 :memory: ,因為這種方式是使用 SQLite in Memory,只要測試一執行完,SQLite in Memory 就會被刪除,因此無法被 Forever 使用,因而無法使用非同步方式寄送 email。

建立 jobs table

oomusou@mac:~/MyProject$ php artisan queue:table
oomusou@mac:~/MyProject$ php artisan migrate

目前是將 queue 建立在資料庫,因此須建立 jobs table。 7 7 GitHub Commit : 建立 jobs table

安裝 Forever

在 Laravel 的 Queues 文件中,是建議大家使用 Supervisor ,它會讓 php artisan queue:listen 在背景執行,持續地消耗 queue,不過 Supervisor 是 Linux 的程式,無法在 Windows 與 macOS 執行,因此對於開發並不方便。

這裡介紹的是由 Node.js 開發的 Forever ,功能與 Supervisor 完全一樣,但因為由 Node.js 所開發,在 Windows、 macOS 與 Linux 都可執行,只要能能成功安裝 Node.js 即可。

oomusou@mac:~/$ npm install -g forever

forever 全域安裝。

使用 Forever 啟動 Mail Queue

oomusou@mac:~/$ forever start -l /Users/oomusou/Code/Laravel/Laravel52QueueForever/forever.log -c php artisan queue:listen
  • start : 使用 forever 啟動其他服務。
  • -l : 指定 log 位置, Forever 預設將 log 放在 ~/.forever 目錄下,且每次啟動為亂數,若你想將 log 指定在特定目錄,並使用特定檔名,則必須使用 -l ,且必須使用完整路徑。
  • -c : 要啟動的 CLI 命令。

由 Queue 寄送信件

MailService.php 8 8 GitHub Commit : 由 queue 寄送信件

tests/MailService.php
namespace App\Services;

use Illuminate\Mail\Message;
use Mail;

class MailService
{
    public function mailByQueue()
    {
        Mail::queue('welcome', [], function (Message $message) {
            $message->sender('oomusou@gmail.com');
            $message->subject('Laravel 5.2 mail by Queue');
            $message->to('oomusou@gmail.com');
        });
    }
}

使用 Mail::queue() 寄送信件,其他參數與 Mail::send() 完全相同。

實際執行發現, Mail::queue() 會馬上執行完,不會有任何等待,因為是非同步,不用等 smtp server 回應就可繼續執行。

Conclusion

  • 由同步改成非同步方式寄信,在程式方面,只要將 Mail::send() 改成 Mail::queue() 即可。
  • Laravel 支援多種 queue,本文使用最簡單的資料庫方式。
  • Forever 為 Node.js 所開發,在 Windows、macOS 與 Linux 都可使用,非常方便。

Sample Code

完整的範例可以在我的 GitHub 上找到。

Reference

Taylor Otwell, Mail

Taylor Otwell, Queues

foreverjs, forever

 
标签: Laravel 数据库
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • nginx 各类网站设置 (laravel , thinkphp , nod
    基础部分设置[root@centos ~]# vim /opt/nginx/conf/nginx.confuser www www;worker_processes auto;pid logs/nginx.pid;worker_rlimit_nofile 100000;events {use epoll;multi_accept on;worker_connections 65535 ;}http {include mime.types;default_type
    02-09
  • PHP trait 特性在 Laravel 中的使用个人心得
    trait 是在PHP5.4中为了方便代码复用的一种实现方式,但目前我在看的的PHP项目中较少看的有程序员去主动使用这个实现方式,在laravel中有很多 trait 的使用,关于trait 在 laravel 的使用请参看 Laravel 在哪些地方用了 trait?我曾在 Laravel 中大型项目面向
    02-09
  • 让我们用 laravel-mix 为 TypeScript 和 Sass
    介绍前端编译TypeScript、Sass、模板引擎等时经常用到Gulp和webpack。这是我个人的印象,但它们似乎都难以管理,因为它们的描述往往复杂而冗长。我不想积极进行,因为我要担心加载器的顺序并且有很多配置选项,我必须花时间去了解它们。我想推荐那里laravel
  • PHP Laravel软删除的实现方法介绍
    用Laravel 自带的 Eloquent ORM 来实现软删除。首先在数据迁移文件中添加删除时间字段./database/migrations/2014_10_12_000000_create_users_table.php?phpuse Illuminate\Database\Migrations\Migration;use Illuminate\Database\Schema\Blueprint;use Illu
  • Laravel中如何使用PHP的装饰器模式 php laravel
    本文小编为大家详细介绍“Laravel中如何使用PHP的装饰器模式”,内容详细,步骤清晰,细节处理妥当,希望这篇“Laravel中如何使用PHP的装饰器模式”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。装饰器模式定义:它可以帮助您在
    02-08 laravelphp
  • PHP laravel使用自定义邮件类实现发送邮件
    PHP laravel使用自定义邮件类实现发送邮件
    当登录邮箱为腾讯企业邮箱的时候。Phpmailer发送邮件就不好用了,具体哪里不好用,我没真没找到。但是,邮件得发啊,怎么办呢?我这里搞了一个自定义的发送邮件类,腾讯企业邮箱也可用。但是,邮件发送失败,不会返回报错信息,这个可能是有点坑。源码如下:?
  • 详解PHP laravel中的加密与解密函数
    目录一:简介二:配置三:使用加密/解密1:加密2:不使用序列化进行加密3:解密Laravel为我们提供了完整的加密方法及加密模式。我之前一般在加密的时候使用的是我自己写的加密函数,但是这个玩意,有的位置还是不太使用,当然,破解的话,基本上也是不可能的
  • PHP laravel缓存cache机制详解
    目录一、访问多个缓存存储二、从缓存中获取数据1.获取数据并设置默认值2.检查缓存项是否存在3.数值增加/减少4.获取存储5.获取删除三、缓存中存储数据1.获取存储数据2.缓存不存在时存储数据3.永久存储数据四、从缓存中移除数据Laravel中的cache为我们提供了三
  • PHP laravel实现导出PDF功能
    PHP laravel实现导出PDF功能
    目录一、laravel-tcpdf二、tcpdf三、TCPDF解决保存中文文件名的方法补充一、laravel-tcpdf导出PDF文件Laravel框架为我们集成了一个插件tcpdf。下载地址:https://github.com/elibyy/tcpdf-laravel然后使用composer进行安装就可以了。具体安装过程,请查看文末
  • PHP laravel缓存cache机制怎么实现
    今天小编给大家分享一下PHP laravel缓存cache机制怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。Laravel中的cache为我们
点击排行