我有一个开发 Web 服务器(CentOS LAMP 堆栈),它使用 postfix 中的 SMTP 中继设置来发送电子邮件。我们将 mailgun 与多个用户一起使用,设置类似于 this ,但针对特定用户而不仅仅是通配符电子邮件:
/etc/postfix/main.cf
smtp_sasl_password_maps = hash:/etc/postfix/sasl_passwd
smtp_sasl_auth_enable = yes
sender_dependent_relayhost_maps = hash:/etc/postfix/relayhost_map
smtp_sender_dependent_authentication = yes
smtp_sasl_security_options = noanonymous
relayhost = [smtp.mailgun.org]:587
/etc/postfix/sasl_passwd
# our domains
no-reply@domain.com postmaster@domain.com:password1
info@domain2.com postmaster@domain2.com:password2
support@domain3.com postmaster@domain3.com:password3
# in-case we don't have it setup, use a default
#[smtp.mailgun.org]:587 postmaster@domain2.com:password2
/etc/postfix/relayhost_map
no-reply@domain.com [smtp.mailgun.org]:587
info@domain2.com [smtp.mailgun.org]:587
support@domain3.com [smtp.mailgun.org]:587
为了设置电子邮件日志记录,我在机器上使用他们自己的 SMTP 凭据对每个开发人员进行身份验证。我想对其进行设置,以便开发人员无需添加 additional_headers 或 additional_parameters 即可在 postfix 中获得正确的 smtp 中继匹配 - 实际上这需要很多时间为不同的开发人员在代码中设置不同的邮件 header 的工作,尤其是版本代码。我离题了。当我使用以下内容时,从 postfix 的角度来看,这工作得很好:
mail('email@address.tld', 'subject', 'message here...', 'From: noreply@domain.com', '-fnoreply@domain.com');
然后我将以下内容添加到虚拟主机配置中:
php_admin_value sendmail_path "/usr/sbin/sendmail -t -i -fnoreply@domain.com"
它成功地让我摆脱了 -f additional_parameter 并且仍然正确发送。然后我添加了以下内容:
php_value sendmail_from "noreply@domain.com"
在 phpinfo() 转储中,我看到 sendmail_from 的本地值设置正确,但是现在当我发送电子邮件时,它显示为:
None@domain.com on behalf of Apache
似乎是正确的发件人(MIME,而不是信封,因为 postfix 识别身份验证并给出 250 Great success)。使用详细的 postfix 日志记录,我只能看到来自发件人输入属性的正确电子邮件的引用。
在 mailgun 中,我从日志中看到以下信息,但是,对于 From: noreply@domain.com 未 使用的电子邮件:
...
"envelope": {
"transport": "smtp",
"sender": "noreply@domain.com",
"sending-ip": "x.x.x.x",
"targets": "email@address.tld"
},
"message": {
"headers": {
"to": "email@address.tld",
"message-id": "2014061111016.ABC1D23456E@domain.com",
"from": "noreply@domain.com (Apache)",
"subject": "Debug Test"
},
"attachments": [],
"recipients": [
"email@address.tld"
],
"size": 654
},
...
有趣的是当 From: noreply@domain.com 出现时,message->headers->from 的日志相同被正确地设置为 noreply@domain.com 而没有添加 (Apache)。当然,这意味着这是 PHP 的错误,并且 PHP 没有正确使用 sendmail_from 值?
考虑到所有这些,我的问题是除了 mail() 函数之外,我如何在 PHP 中设置默认的 MIME 发件人(来自 header )? 我在上面的方法/配置中遗漏了什么,还是根本不可能?我很高兴跳出框框思考一下,因为这将有助于节省我们想要此功能的时间。
最佳答案
来自 http://www.php.net/manual/en/mail.configuration.php :
sendmail_from string
Which "From:" mail address should be used in mail sent from PHP under Windows. This directive also sets the "Return-Path:" header.
因此,因为您使用的是 CentOS LAMP 堆栈,不是 Windows,所以此设置将被忽略。
还有:
sendmail_path string
If set,smtp,smtp_portandsendmail_fromare ignored and the specified command is executed.
所以 sendmail_from也被忽略,因为您正在使用 sendmail_path .
您还可以传递 -F (大写 F)用空值从 header 中删除“发件人全名”:
通过 -f noreply@domain.com -F ""到 $additional_parameters mail() 的参数功能。
当您看到您的发件人地址发生变化时,Postfix 正在这样做。这是因为(在您的情况下)运行 Apache 的系统用户正在调用 sendmail二进制。这个二进制文件将使用 <system-user>@<hostname>作为发件人地址(除非 -f 或 -r 参数另有说明)。
为了防止这种情况,sendmail应该用 -f <sender address> 调用.通过将其传递给 $additional_parameters mail() 的参数功能,或将其添加到 sendmail_path ini 设置。
但看起来你已经想通了:)
我建议您根本不要使用 ini 设置。相反,围绕 mail() 编写一个小类功能:
class Mailer
{
/**
* @var string
*/
private $fromEmail;
/**
* @var string
*/
private $fromName;
/**
* @param string $fromEmail
* @param string $fromName
*/
public function __construct($fromEmail, $fromName)
{
$this->fromEmail = $fromEmail;
$this->fromName = $fromName;
}
/**
* @param string $to
* @param string $subject
* @param string $body
* @param array $headers
* @return bool
*/
public function send($to, $subject, $body, array $headers = array())
{
$headers[] = sprintf('From: "%s" <%s>', $this->fromName, $this->fromEmail);
$parameters = sprintf('-f %s', $this->fromEmail);
return mail($to, $subject, $body, $headers, $parameters);
}
}
然后像这样使用它:
$mailer = new Mailer('noreply@domain.com', 'My Company');
$mailer->send('email@address.tld', 'Subject', 'Body');
这门课过于简单化了,但它应该让您基本了解如何消除 From: 的麻烦。 header 和 -f每次要发送电子邮件时的参数。
如果您使用依赖注入(inject)容器或注册表,您可以在应用程序的引导阶段配置/实例化此类,并在您想发送电子邮件时从容器/注册表中获取它:
$container->get('mailer')->send($to, $subject, $message);
或者使用注册表:
Registry::get('mailer')->send($to, $subject, $message);
从长远来看,研究第 3 方库来发送电子邮件是值得的。我个人比较喜欢Swift Mailer .
关于php - 我怎样才能强制从 : header (envelope sender) in PHP without putting it in mail()?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24183058/
如果我使用ruby版本2.5.1和Rails版本2.3.18会怎样?我有基于rails2.3.18和ruby1.9.2p320构建的rails应用程序,我只想升级ruby的版本,而不是rails,这可能吗?我必须面对哪些挑战? 最佳答案 GitHub维护apublicfork它有针对旧Rails版本的分支,有各种变化,它们一直在运行。有一段时间,他们在较新的Ruby版本上运行较旧的Rails版本,而不是最初支持的版本,因此您可能会发现一些关于需要向后移植的有用提示。不过,他们现在已经有几年没有使用2.3了,所以充其量只能让更
其实做自媒体的成本并不高,入门只需要一部手机即可!在手机上找视频素材、使用手机剪辑视频、最后使用手机发布视频作品获得收益!方法并不难,今天这期内容就来给粉丝们分享一种小方法,每天稳定收益100-300,抓紧点赞收藏!1、找素材(1)使用手机拍摄自己喜欢的经典段落,使用程序把文案内容提取出来(2)也可以在豆瓣、知乎、微博等网站中找一些自己需要的文案素材(3)把文案进行润色修改,可以加入一些自己的观点(4)视频素材可以使用软件中自带的素材,也可以在素材网站中下载完整版的素材2、文案配音(1)把复制好的文案直接导入小程序中(2)调整音色、音调后一键合成音频即可(3)可以选择自己朗读配音,需要花一点时
我正在使用Heroku(heroku.com)来部署我的Rails应用程序,并且正在构建一个iPhone客户端来与之交互。我的目的是将手机的唯一设备标识符作为HTTPheader传递给应用程序以进行身份验证。当我在本地测试时,我的header通过得很好,但在Heroku上它似乎去掉了我的自定义header。我用ruby脚本验证:url=URI.parse('http://#{myapp}.heroku.com/')#url=URI.parse('http://localhost:3000/')req=Net::HTTP::Post.new(url.path)#boguspara
-if!request.path_info.include?'A'%{:id=>'A'}"Text"-else"Text"“文本”写了两次。我怎样才能只写一次并同时检查path_info是否包含“A”? 最佳答案 有两种方法可以做到这一点。使用部分,或使用content_forblock:如果“文本”较长,或者是一个重要的子树,您可以将其提取到一个部分。这会使您的代码变干一点。在给出的示例中,这似乎有点矫枉过正。在这种情况下更好的方法是使用content_forblock,如下所示:-if!request.path_info.inc
我有这个代码:context"Visitingtheusers#indexpage."dobefore(:each){visitusers_path}subject{page}pending('iii'){shouldhave_no_css('table#users')}pending{shouldhavecontent('Youhavereachedthispageduetoapermissionic错误')}它会导致几个待处理,例如ManagingUsersGivenapractitionerloggedin.Visitingtheusers#indexpage.#Noreason
考虑这个,它工作正常::>.to_proc.curry(2)[9][8]#=>true,because9>8然而,即使>是一个二元运算符,如果没有指定的元数,上面的代码将无法工作::>.to_proc.curry[9][8]#=>ArgumentError:wrongnumberofarguments(0for1)为什么两者不等价?注意:我特别想用提供的一个参数创建中间柯里化(Currying)函数,然后然后调用然后用第二个参数调用它。 最佳答案 curry必须知道传入的过程的数量,对吧?:-1来自arity的负值令人困惑,但基本上
我要下载http://foobar.com/song.mp3作为song.mp3,而不是让Chrome在其native中打开它浏览器中的播放器。我怎样才能做到这一点? 最佳答案 您只需要确保发送这些header:Content-Disposition:attachment;filename=song.mp3;Content-Type:application/octet-streamContent-Transfer-Encoding:binarysend_file方法为您完成:get'/:file'do|file|file=File.
我正在为我的用户实现一些rubyonrails代码推特内容。我正在创建正确的oauth链接...类似http://twitter.com/oauth/authorize?oauth_token=y2RkuftYAEkbEuIF7zKMuzWN30O2XxM8U9j0egtzKv但在我的测试帐户授予对twitter的访问权限后,它会弹出一个页面,上面写着“您已成功授予对.我不知道用户应该在哪里输入此PIN以及他们为什么必须这样做。我认为这不是必要的步骤。Twitter应该将用户重定向到我在应用程序设置中提供的回调URL。有谁知道为什么会这样?更新我找到了thisarticle声明我需
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭9年前。我最近开始学习Ruby,这是我的第一门编程语言。我对语法感到满意,并且我已经完成了许多只教授相同基础知识的教程。我已经写了一些小程序(包括我自己的数组排序方法,在有人告诉我谷歌“冒泡排序”之前我认为它非常聪明),但我觉得我需要尝试更大更难的东西来理解更多关于Ruby.关于如何执行此操作的任何想法?
我想在我的Controller中使用以下corsheader呈现JSON:'Access-Control-Allow-Origin'='*'.我试过这个:defmy_actionrender(json:some_params)response.headers['Access-Control-Allow-Origin']='*'end但是我得到了一个AbstractController::DoubleRenderError。有没有办法使用header呈现JSON? 最佳答案 您不能在渲染后设置header,因为已发送响应。所以在没有意