我正在使用Laravel 4的Mail::queue()
使用内置的Mailgun驱动程序发送电子邮件。问题是我希望能够发送多个Mailgun域名,但必须在app/config/services.php
中设置域名。由于我使用Mail::queue()
,我无法看到如何动态设置该配置变量。
有什么方法可以做我要问的事吗?理想情况下,当我调用Mail::queue()
时,我希望能够传入域名(Mailgun api密钥对于我想发送的所有域名都是相同的)。
答案 0 :(得分:3)
这适用于Laravel 5.4:
// Get the existing SwiftMailer
$swiftMailer = Mail::getSwiftMailer();
// Update the domain in the transporter (Mailgun)
$transport = $swiftMailer->getTransport();
$transport->setDomain('YOUR-DOMAIN.HERE');
// Use the updated version
$mailer = Swift_Mailer::newInstance($transport);
Mail::setSwiftMailer($mailer);
答案 1 :(得分:2)
在运行时切换Laravel Mailer的配置细节并不难,但我不知道使用Mail::queue
外观可以做任何事情。可以使用Queue::push
和Mail::send
的组合来完成(无论如何Mail::queue
都会这样做。)
Mail::queue
外观的问题是传递给闭包的$message
参数是Illuminate\Mail\Message
类型,我们需要修改邮件传输,只能通过Swift_Mailer
实例(并且只在Message
类中读取)。
您需要使用使用所需域的Mailgun传输实例创建一个负责发送电子邮件的类:
use Illuminate\Mail\Transport\MailgunTransport;
use Illuminate\Support\SerializableClosure;
class SendQueuedMail {
public function fire($job, $params)
{
// Get the needed parameters
list($domain, $view, $data, $callback) = $params;
// Backup your default mailer
$backup = Mail::getSwiftMailer();
// Setup your mailgun transport
$transport = new MailgunTransport(Config::get('services.mailgun.secret'), $domain);
$mailer = new Swift_Mailer($transport);
// Set the new mailer with the domain
Mail::setSwiftMailer($mailer);
// Send your message
Mail::send($view, $data, unserialize($callback)->getClosure());
// Restore the default mailer instance
Mail::setSwiftMailer($backup);
}
}
现在您可以对这样的电子邮件进行排队:
use Illuminate\Support\SerializableClosure;
...
Queue::push('SendQueuedMail', ['domain.com', 'view', $data, serialize(new SerializableClosure(function ($message)
{
// do your email sending stuff here
}))]);
虽然它没有使用Mail::queue
,但这种替代方案同样紧凑且易于阅读。此代码未经过测试但应该可以使用。
答案 2 :(得分:2)
我使用Macros
添加动态配置。我不记得这是否可以在Laravel 4中完成,但可以在5上完成。
在服务提供商(AppServiceProvider
)
public function boot()
{
Mail::macro('setConfig', function (string $key, string $domain) {
$transport = $this->getSwiftMailer()->getTransport();
$transport->setKey($key);
$transport->setDomain($domain);
return $this;
});
}
然后我可以这样使用:
\Mail::setConfig($mailgunKey, $mailgunDomain)->to(...)->send(...)
在你的情况下
\Mail::setConfig($mailgunKey, $mailgunDomain)->to(...)->queue(...)
答案 3 :(得分:0)
我的用例与此类似,简而言之,我只是想通过查看消息的 from 地址字段中设置的域(我在运行时自动配置mailgun发送域)在使用Mail::from(...)->send(...)
发送之前进行即时设置)。如果他们正在设置邮件中的发件人地址以匹配mailgun发送域,则可以解决OP的用例。
我的解决方案注册了一个替代MailgunTransport,它将覆盖内置的MailgunTransport,并在发送前设置域。这样,我只需要在我的mail.php
中注册新的驱动程序,然后调用Mail::send
或Mail::queue
。
config \ mail.php:
'driver' => env('MAIL_DRIVER', 'mailgun-magic-domain')
providers \ MailgunMagicDomainProvider:
<?php
namespace App\Providers;
use Illuminate\Support\ServiceProvider;
use Illuminate\Mail\Transport\MailgunTransport;
use Swift_Mime_Message;
use Illuminate\Support\Arr;
use GuzzleHttp\Client as HttpClient;
class MailgunMagicDomainProvider extends ServiceProvider
{
/**
* Bootstrap the application services.
*
* @return void
*/
public function boot()
{
$swiftTransport = $this->app['swift.transport'];
$swiftTransport->extend('mailgun-magic-domain', function($app) {
$config = $app['config']->get('services.mailgun', []);
$client = new HttpClient(Arr::add(
Arr::get($config, 'guzzle', []), 'connect_timeout', 60
));
return new MailgunTransportWithDomainFromMessage(
$client,
$config['secret'],
$config['domain'] // <- we have to pass this in to avoid re-writing the whole transport, but we'll be dynamically setting this before each send anyway
);
});
}
/**
* Register the application services.
*
* @return void
*/
public function register()
{
}
}
/**
* Overrides the built in Illuminate\Mail\Transport\MailgunTransport but doesnt pull the
* mailgun sending domain from the config, instead it uses the domain in the from address
* to dynamically set the mailgun sending domain
*/
class MailgunTransportWithDomainFromMessage extends MailgunTransport
{
/**
* {@inheritdoc}
*/
public function send(Swift_Mime_Message $message, &$failedRecipients = null)
{
$this->setDomain($this->getDomainFromMessage($message));
return parent::send($message, $failedRecipients);
}
protected function getDomainFromMessage(Swift_Mime_Message $message)
{
$fromArray = $message->getFrom();
if (count($fromArray) !== 1) {
throw new \Exception('Cannot use the mailgun-magic-domain driver when there isn\'t exactly one from address');
}
return explode('@', array_keys($fromArray)[0])[1];
}
}
config / app.php:
'providers' => [
...
\App\Providers\MailgunMagicDomainProvider::class
],
答案 4 :(得分:0)
也许对某人有用,我如下解决;
在启动功能/方法下的 ServiceProvider
中;
public function boot()
{
Mail::macro('setConfig', function (string $key, string $domain) {
config()->set('services', array_merge(config('services'), [
'mailgun' => [
'domain' => $domain,
'secret' => $key
]
]));
});
}
呼叫排队
Mail::setConfig($key, $domain)->to(...)->queue(...)