Discuz开发 Discuz论坛发送邮件原理问题分析

[复制链接]
ARCHY明星会员实名认证 发表于 2012-5-23 23:28:03 [Discuz开发] 显示全部楼层 |阅读模式 上一主题 下一主题
ad广告
ad广告

马上注册,一起探讨正确快速的建站方法

您需要 登录 才可以下载或查看,没有帐号?快速注册

x
这里就针对dx的发送邮件函数进行一个简单的分析,供人参考,如果知道发生原理了还不能帮助你解决你的discuz邮件发送问题,可以看看Discuz! 邮件发送失败排查思路

推荐视频教程: Discuz 注册设置和邮箱设置

下面开始。

邮件发送常用的有2种方式
1、sendmail方式,这种方式要服务器支持,不需要提供其他。
2、socket链接smtp方式,这种方式基本上服务器都可以使用这种方式,但是需要你提供smtp服务器 用户名 密码等。

发送邮件不成功的话,去检查data/log 下的 xxx__SMTP.php文件内容XXX是本月的日期,发送邮件的过程中出现的问题,也都会以日志的方式保存在这里。这个日志里面基本上记录的都是第二种方式出现的错误。

以下并非完全分析发送邮件函数,只检出来对大家可能有用的说。

1、sendmail方式,这种方式要服务器支持,不需要提供其他。
  1. if(function_exists('mail') && @mail($email_to, $email_subject, $email_message, $headers)) {
  2.                         return true;
  3.                 }
  4.                 return false;
复制代码
判断mail函数是否存在和使用mail函数发信。失败则返回false。sendmail函数并不在log文件中输出错误信息。不过一般服务器sendmail没有问题的话,都能发出去了。

2、socket链接smtp方式
首先用下面代码建立链接,如果失败则记录日志 XXX CONNECT - Unable to connect to the SMTP server.这个时候你就要检查你服务器与smtp服务器之间建立链接的时候出现了什么问题。
  1. if(!$fp = fsockopen($_G['setting']['mail']['server'], $_G['setting']['mail']['port'], $errno, $errstr, 30)) {
  2.                         runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) CONNECT - Unable to connect to the SMTP server", 0);
  3.                         return false;
  4.                 }
复制代码
下面代码从建立的连接中读取512个字符,如果读取到的前三个字符不等于220那么则输出错误信息XXX_ CONNECT xxx
用qqmail 来举例这里得到的正常值应该是类似这样一句 220 esmtp10.qq.com Esmtp QQ Mail Server
  1. $lastmessage = fgets($fp, 512);
  2.                 if(substr($lastmessage, 0, 3) != '220') {
  3.                         runlog('SMTP', "{$_G[setting][mail][server]}:{$_G[setting][mail][port]} CONNECT - $lastmessage", 0);
  4.                         return false;
  5.                 }
复制代码
下面代码是向邮件服务器标示用户身份。如果后台填写smtp服务器那块勾上验证的勾,就发送 EHLO 反之,发送 HELO。
然后再读512个字符,判断是否是220或者250.否的话则输出错误信息。
  1. fputs($fp, ($_G['setting']['mail']['auth'] ? 'EHLO' : 'HELO')." uchome\r\n");
  2.                 $lastmessage = fgets($fp, 512);
  3.                 if(substr($lastmessage, 0, 3) != 220 && substr($lastmessage, 0, 3) != 250) {
  4.                         runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) HELO/EHLO - $lastmessage", 0);
  5.                         return false;
  6.                 }

  7.                 while(1) {
  8.                         if(substr($lastmessage, 3, 1) != '-' || empty($lastmessage)) {
  9.                                 break;
  10.                         }
  11.                         $lastmessage = fgets($fp, 512);
  12.                 }
复制代码
如果前面勾上需要验证的话,后面就要进行验证的步骤了
先发送 AUTH LOGIN\r\n  然后取消息,如果返回消息开头不等于334 则记录错误。
然后发送SMTP 身份验证用户名。并判断是否错误。
最后发送 SMTP 身份验证密码。并判断是否错误。
在这里顺便说一句,本人在这里使用QQ邮箱做测试。在QQ邮箱中有一项如果不设置,得到的会是 ”454 Authentication failed, please open smtp flag first! “
  1.                 if($_G['setting']['mail']['auth']) {
  2.                         fputs($fp, "AUTH LOGIN\r\n");
  3.                         $lastmessage = fgets($fp, 512);
  4.                         echo "<br/>".$lastmessage."<br/>";
  5.                         if(substr($lastmessage, 0, 3) != 334) {
  6.                                 runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) AUTH LOGIN - $lastmessage", 0);
  7.                         }

  8.                         fputs($fp, base64_encode($_G['setting']['mail']['auth_username'])."\r\n");
  9.                         $lastmessage = fgets($fp, 512);
  10.                         if(substr($lastmessage, 0, 3) != 334) {
  11.                                 runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) USERNAME - $lastmessage", 0);
  12.                                 echo "$lastmessage:".$lastmessage."<br/>";
  13.                         }

  14.                         fputs($fp, base64_encode($_G['setting']['mail']['auth_password'])."\r\n");
  15.                         $lastmessage = fgets($fp, 512);
  16.                         if(substr($lastmessage, 0, 3) != 235) {
  17.                                 runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) PASSWORD - $lastmessage", 0);
  18.                                 echo "e";
  19.                         }

  20.                         $email_from = $_G['setting']['mail']['from'];
  21.                 }
复制代码
后面的一起来说,前面链接成功了,验证成功了。后面就是发送邮件主体的部分了。没一步发送信息后都要对服务器返回值进行判断,如果开头不等于250 均记录错误日志并且退出。
  1. fputs($fp, "MAIL FROM: <".preg_replace("/.*\<(.+?)\>.*/", "\\1", $email_from).">\r\n");
  2.                 $lastmessage = fgets($fp, 512);
  3.                 if(substr($lastmessage, 0, 3) != 250) {
  4.                         fputs($fp, "MAIL FROM: <".preg_replace("/.*\<(.+?)\>.*/", "\\1", $email_from).">\r\n");
  5.                         $lastmessage = fgets($fp, 512);
  6.                         if(substr($lastmessage, 0, 3) != 250) {
  7.                                 runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) MAIL FROM - $lastmessage", 0);
  8.                                 return false;
  9.                         }
  10.                 }

  11.                 fputs($fp, "RCPT TO: <".preg_replace("/.*\<(.+?)\>.*/", "\\1", $toemail).">\r\n");
  12.                 $lastmessage = fgets($fp, 512);
  13.                 if(substr($lastmessage, 0, 3) != 250) {
  14.                         fputs($fp, "RCPT TO: <".preg_replace("/.*\<(.+?)\>.*/", "\\1", $toemail).">\r\n");
  15.                         $lastmessage = fgets($fp, 512);
  16.                         runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) RCPT TO - $lastmessage", 0);
  17.                         return false;
  18.                 }

  19.                 fputs($fp, "DATA\r\n");
  20.                 $lastmessage = fgets($fp, 512);
  21.                 if(substr($lastmessage, 0, 3) != 354) {
  22.                         runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) DATA - $lastmessage", 0);
  23.                         return false;
  24.                 }

  25.                 $headers .= 'Message-ID: <'.gmdate('YmdHs').'.'.substr(md5($email_message.microtime()), 0, 6).rand(100000, 999999).'@'.$_SERVER['HTTP_HOST'].">{$maildelimiter}";

  26.                 fputs($fp, "Date: ".gmdate('r')."\r\n");
  27.                 fputs($fp, "To: ".$email_to."\r\n");
  28.                 fputs($fp, "Subject: ".$email_subject."\r\n");
  29.                 fputs($fp, $headers."\r\n");
  30.                 fputs($fp, "\r\n\r\n");
  31.                 fputs($fp, "$email_message\r\n.\r\n");
  32.                 $lastmessage = fgets($fp, 512);
  33.                 if(substr($lastmessage, 0, 3) != 250) {
  34.                         runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) END - $lastmessage", 0);
  35.                 }
  36.                 fputs($fp, "QUIT\r\n");
复制代码
如果说我不想看上面的,我就想查找我到底是哪一步出现问题了。那么我把发送邮件函数摘出来了。并且每一步输出了文字信息在浏览器上。这样您拿着文字信息来找人帮你分析是什么错误。

首先在后台设置好邮件设置

然后下载附件中的testmail.zip 解压缩。

修改里面的 testmail.php 编辑三项
$toemail = "";   #发送到XXX邮箱,填写您要发送到的邮箱。
$subject = "";    #邮件主体
$message = "";    #邮件内容
在引号之中填写信息。

把 testmail.php 上传到你网站的根目录

然后执行
例如 www.xxx.com/testmail.php

看输出的信息来判断问题。

最后解决了之后不要忘记删除这个文件。

推荐相关阅读:

0、Discuz X3.2/x2.5 后台邮件设置方法 - QQ企业邮箱为例

1、Discuz X3.2/x2.5 后台邮件设置方法 以QQ邮箱为例子

2、Discuz! X3.2/x2.5 后台邮件配置详解--使用163邮箱




上一篇:Discuz X2 后台邮件设置方法 以QQ邮箱为例子
下一篇:Discuz论坛邮件不能发送的原因和解决办法

大神点评13

非一般感觉明星会员实名认证 发表于 2012-5-25 11:37:50 [Discuz开发] 显示全部楼层
了解下了
hack_小灰 发表于 2012-7-24 23:03:38 [Discuz开发] 显示全部楼层
附件呢? 附件在哪?
fengying 发表于 2013-10-14 21:21:47 [Discuz开发] 显示全部楼层
你这里真是太棒了,解决问题比官方网还要专用,还要认真,非常感谢你!

不过我还是暂时没有解决这个问题,相信有了你的帖子,我会把问题解决掉的。

再次感谢!
314159 发表于 2013-11-6 10:36:11 [Discuz开发] 显示全部楼层
您好    我想在论坛帖子详情页弄一个邮件按钮    点击按钮填写相关的资料之后可以将帖子内容发送出去   请问要怎么做呢   简单的提示一下就好了   我用mail函数和   sendmail函数发送测试邮件都没有成功    做的是本地测试    系统是win7   64位    环境是appserver 谢谢
v4521 发表于 2014-4-6 16:20:50 [Discuz开发] 显示全部楼层
说的非常好啊,很具体。顶zb7论坛!
1981819858 发表于 2016-4-5 14:02:04 [Discuz开发] 显示全部楼层
附件在哪?
chump90 发表于 2016-12-27 11:22:17 [Discuz开发] 显示全部楼层
强烈支持,站帮网有你更精彩
chump90 发表于 2016-12-27 11:23:11 [Discuz开发] 显示全部楼层
强烈支持,站帮网有你更精彩
T18218967676 发表于 2017-6-5 16:56:46 [Discuz开发] 显示全部楼层
附件在哪里??????
haonvhi 发表于 2018-2-27 21:19:15 [Discuz开发] 显示全部楼层
找了这么久,在这里被找到了一直z在用这个模板
淘小淘 发表于 2019-6-3 10:19:23 [Discuz开发] 显示全部楼层
回帖支持下,期待更多分享。
您需要登录后才可以回帖 登录 | 快速注册

本版积分规则

快速回复 返回顶部 返回列表