制作PHP';s mail()异步


Making PHP's mail() asynchronous

我有使用ssmtp的PHP的mail(),它没有队列/假脱机,并且与AWS SES同步。

我听说我可以使用SwiftMail提供一个假脱机,但我无法像目前使用mail()那样制定出一个简单的使用方法。

我想要最少的代码量来提供异步邮件。我不在乎电子邮件是否发送失败,但最好有一个日志。

有什么简单的技巧或窍门吗?缺少一个完整的邮件服务器?我想sendmail包装器可能是答案,但我无法计算出nohup

有很多方法可以做到这一点,但处理线程不一定是正确的选择。

  • register_shutdown_function:在发送响应后调用关闭函数。它并不是真正的异步,但至少不会减慢您的请求。关于实施,请参见示例
  • Swift池:使用symfony,可以轻松使用spool
  • Queue:在队列系统中注册要发送的邮件(可以使用RabbitMQ、MySQL、redis或任何东西),然后运行消耗队列的cron。可以使用MySQL表这样简单的东西来完成,该表包含fromtomessagesent等字段(当您发送电子邮件时,布尔值设置为true

寄存器_关闭_函数示例

<?php
class MailSpool
{
  public static $mails = [];
  public static function addMail($subject, $to, $message)
  {
    self::$mails[] = [ 'subject' => $subject, 'to' => $to, 'message' => $message ];
  }
  public static function send() 
  {
    foreach(self::$mails as $mail) {
      mail($mail['to'], $mail['subject'], $mail['message']);
    }
  }
}
//In your script you can call anywhere
MailSpool::addMail('Hello', 'contact@example.com', 'Hello from the spool');

register_shutdown_function('MailSpool::send');
exit(); // You need to call this to send the response immediately

php fpm

您必须运行php-fpm才能使用fastcgi_finish_request。

echo "I get output instantly";
fastcgi_finish_request(); // Close and flush the connection.
sleep(10); // For illustrative purposes. Delete me.
mail("test@example.org", "lol", "Hi");

在完成对用户的请求后,很容易排队处理任意代码:

$post_processing = [];
/* your code */
$email = "test@example.org";
$subject = "lol";
$message = "Hi";
$post_processing[] = function() use ($email, $subject, $message) {
  mail($email, $subject, $message);
};
echo "Stuff is going to happen.";
/* end */
fastcgi_finish_request();
foreach($post_processing as $function) {
  $function();
}

嬉皮士背景工作者

立即暂停一个curl,让新的请求来处理它。我在酷之前就在共享主机上这样做了(从来都不酷)

if(!empty($_POST)) {
  sleep(10);
  mail($_POST['email'], $_POST['subject'], $_POST['message']);
  exit(); // Stop so we don't self DDOS.
}
$ch = curl_init("http://" . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI']);
curl_setopt($ch, CURLOPT_TIMEOUT, 1);
curl_setopt($ch, CURLOPT_NOSIGNAL, 1);
curl_setopt($ch, CURLOPT_POSTFIELDS, [
  'email' => 'noreply@example.org',
  'subject' => 'foo',
  'message' => 'bar'
]);
curl_exec($ch);
curl_close($ch);
echo "Expect an email in 10 seconds.";

将AWS SES与PHPMailer一起使用。

这种方式非常快(每秒数百条消息),而且不需要太多代码。

$mail = new PHPMailer;
$mail->isSMTP();                                      // Set mailer to use SMTP
$mail->Host = 'ssl://email-smtp.us-west-2.amazonaws.com';  // Specify main and backup SMTP servers
$mail->SMTPAuth = true;                               // Enable SMTP authentication
$mail->Username = 'blah';                 // SMTP username
$mail->Password = 'blahblah';                           // SMTP password

$mail->SMTPSecure = 'tls';                            // Enable TLS encryption, `ssl` also accepted
$mail->Port = 443; 

不确定我是否正确解释了你的问题,但我希望这能有所帮助。

Pthreads是您的朋友:)
这是我如何在生产应用程序中制作的一个示例

class AsynchMail extends Thread{
    private $_mail_from;
    private $_mail_to;
    private $_subject;
    public function __construct($subject, $mail_to, ...) {
        $this->_subject = $subject;
        $this->_mail_to = $mail_to;
        // ... 
    }
    // ...
    // you must redefine run() method, and to execute it we must call start() method
    public function run() {
        // here put your mail() function
        mail($this->_mail_to, ...);
    }
}

测试脚本示例

$mail_to_list = array('Shigeru.Miyamoto@nintendo.com', 'Eikichi.Kawasaki@neogeo.com',...);
foreach($mail_to_list as $mail_to) {
    $asynchMail = new AsynchMail($mail_to);
    $asynchMail->start();
}

如果您需要在PHP中安装和使用线程的进一步帮助,请告诉我
对于日志系统,我强烈建议您使用Log4PHP:功能强大,易于使用和配置
对于发送邮件,我还强烈建议您使用PHPMailer

我使用beanstalkd执行异步php
它是一个简单的消息队列,非常轻量级,易于集成。

使用以下php包装器https://github.com/pda/pheanstalk您可以执行以下操作来实现电子邮件工作程序:

use Beanstalk'Client;
$msg="dest_email##email_subject##from_email##email_body";
$beanstalk = new Client(); 
$beanstalk->connect();
$beanstalk->useTube('flux'); // Begin to use tube `'flux'`.
$beanstalk->put(
    23,  // Give the job a priority of 23.
    0,   // Do not wait to put job into the ready queue.
    60,  // Give the job 1 minute to run.
    $msg // job body
);
$beanstalk->disconnect();

然后,这项工作将在一个单独的php文件中的代码中完成
类似于:

use Beanstalk'Client;
$do=true;
try {
    $beanstalk = new Client();
    $beanstalk->connect();
    $beanstalk->watch('flux');
} catch (Exception $e ) {
    echo $e->getMessage();
    echo $e->getTraceAsString();
    $do = false;
}
while ($do) {
    $job = $beanstalk->reserve(); // Block until job is available.
    $emailParts = explode("##", $job['body'] );
    // Use your SendMail function here
    if ($i_am_ok) {
        $beanstalk->delete($job['id']);
    } else {
        $beanstalk->bury($job['id'], 20);
    }
}
$beanstalk->disconnect();

您可以单独运行这个php文件,作为一个独立的php进程。假设您将其保存为sender.php,它将在Unix中运行为:

php /path/to/sender/sender.php & && disown

该命令将运行该文件,还允许您在不停止进程的情况下关闭控制台或注销当前用户
还要确保您的web服务器使用与php命令行解释器相同的php.ini文件。(可以使用您最喜欢的php.ini链接来解决

我希望它能有所帮助。

您的最佳选择是堆叠或后台打印模式。它相当简单,可以用两个步骤来描述。

  • 将您的电子邮件存储在当前线程上带有已发送标志的表中
  • 使用cron或ajax反复调用一个邮件处理php文件,该文件将从数据库中获取前10或20封未发送的电子邮件,将它们标记为已发送,并通过您最喜欢的邮寄方法实际发送

一个简单的方法是异步调用处理邮件的代码。

例如,如果您有一个名为email.php的文件,其代码如下:

// Example array with e-mailaddresses
$emailaddresses = ['example1@test.com', 'example2@example.com', 'example1@example.com'];
// Call your mail function
mailer::sendMail($emailaddresses);

然后,您可以在像这样的正常请求中异步调用它

exec('nice -n 20 php email.php > /dev/null & echo $!');

请求将在不等待email.php完成发送电子邮件的情况下完成。日志记录也可以添加到处理电子邮件的文件中。

变量可以在被调用的文件名和类似的> /dev/null之间传递到exec中

exec('nice -n 20 php email.php '.$var1.' '.$var2.' > /dev/null & echo $!');

使用escapestellarg()确保这些变量是安全的。在被调用的文件中,这些变量可以与$argv 一起使用

欢迎使用异步PHPhttps://github.com/shuchkin/react-smtp-client

$loop = 'React'EventLoop'Factory::create();
$smtp = new 'Shuchkin'ReactSMTP'Client( $loop, 'tls://smtp.google.com:465', 'username@gmail.com','password' );
$smtp->send('username@gmail.com', 'sergey.shuchkin@gmail.com', 'Test ReactPHP mailer', 'Hello, Sergey!')->then(
    function() {
        echo 'Message sent via Google SMTP'.PHP_EOL;
    },
    function ( 'Exception $ex ) {
        echo 'SMTP error '.$ex->getCode().' '.$ex->getMessage().PHP_EOL;
    }
);
$loop->run();