本帖最后由 rikioy 于 2012-3-14 10:55 编辑
最近经常接到电话或者在论坛上看到有人询问,我在后台设置了,却发不了邮件,我以前能发邮件,升级之后发不了邮件等等问题。
我在这里就针对dx的发送邮件函数进行一个简单的分析,供人参考,如果说我不想看这个分析,我想查找我的问题。我不懂你说的是什么。那么请跳到二楼。
下面开始。
邮件发送常用的有2种方式
1、sendmail方式,这种方式要服务器支持,不需要提供其他。
2、socket链接smtp方式,这种方式基本上服务器都可以使用这种方式,但是需要你提供smtp服务器 用户名 密码等。
发送邮件不成功的话,去检查data/log 下的 xxx__SMTP.php文件内容XXX是本月的日期,发送邮件的过程中出现的问题,也都会以日志的方式保存在这里。这个日志里面基本上记录的都是第二种方式出现的错误。
以下并非完全分析发送邮件函数,只检出来对大家可能有用的说。
1、sendmail方式,这种方式要服务器支持,不需要提供其他。
- if(function_exists('mail') && @mail($email_to, $email_subject, $email_message, $headers)) {
- return true;
- }
- return false;
复制代码 判断mail函数是否存在和使用mail函数发信。失败则返回false。sendmail函数并不在log文件中输出错误信息。不过一般服务器sendmail没有问题的话,都能发出去了。
2、socket链接smtp方式
首先用下面代码建立链接,如果失败则记录日志 XXX CONNECT - Unable to connect to the SMTP server.这个时候你就要检查你服务器与smtp服务器之间建立链接的时候出现了什么问题。- if(!$fp = fsockopen($_G['setting']['mail']['server'], $_G['setting']['mail']['port'], $errno, $errstr, 30)) {
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) CONNECT - Unable to connect to the SMTP server", 0);
- return false;
- }
复制代码
下面代码从建立的连接中读取512个字符,如果读取到的前三个字符不等于220那么则输出错误信息XXX_ CONNECT xxx
用qqmail 来举例这里得到的正常值应该是类似这样一句 220 esmtp10.qq.com Esmtp QQ Mail Server- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != '220') {
- runlog('SMTP', "{$_G[setting][mail][server]}:{$_G[setting][mail][port]} CONNECT - $lastmessage", 0);
- return false;
- }
复制代码
下面代码是向邮件服务器标示用户身份。如果后台填写smtp服务器那块勾上验证的勾,就发送 EHLO 反之,发送 HELO。
然后再读512个字符,判断是否是220或者250.否的话则输出错误信息。
- fputs($fp, ($_G['setting']['mail']['auth'] ? 'EHLO' : 'HELO')." uchome\r\n");
- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != 220 && substr($lastmessage, 0, 3) != 250) {
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) HELO/EHLO - $lastmessage", 0);
- return false;
- }
- while(1) {
- if(substr($lastmessage, 3, 1) != '-' || empty($lastmessage)) {
- break;
- }
- $lastmessage = fgets($fp, 512);
- }
复制代码
如果前面勾上需要验证的话,后面就要进行验证的步骤了
先发送 AUTH LOGIN\r\n 然后取消息,如果返回消息开头不等于334 则记录错误。
然后发送SMTP 身份验证用户名。并判断是否错误。
最后发送 SMTP 身份验证密码。并判断是否错误。
在这里顺便说一句,本人在这里使用QQ邮箱做测试。在QQ邮箱中有一项如果不设置,得到的会是 ”454 Authentication failed, please open smtp flag first! “
- if($_G['setting']['mail']['auth']) {
- fputs($fp, "AUTH LOGIN\r\n");
- $lastmessage = fgets($fp, 512);
- echo "<br/>".$lastmessage."<br/>";
- if(substr($lastmessage, 0, 3) != 334) {
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) AUTH LOGIN - $lastmessage", 0);
- }
- fputs($fp, base64_encode($_G['setting']['mail']['auth_username'])."\r\n");
- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != 334) {
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) USERNAME - $lastmessage", 0);
- echo "$lastmessage:".$lastmessage."<br/>";
- }
- fputs($fp, base64_encode($_G['setting']['mail']['auth_password'])."\r\n");
- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != 235) {
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) PASSWORD - $lastmessage", 0);
- echo "e";
- }
- $email_from = $_G['setting']['mail']['from'];
- }
复制代码
后面的一起来说,前面链接成功了,验证成功了。后面就是发送邮件主体的部分了。没一步发送信息后都要对服务器返回值进行判断,如果开头不等于250 均记录错误日志并且退出。
- fputs($fp, "MAIL FROM: <".preg_replace("/.*\<(.+?)\>.*/", "\\1", $email_from).">\r\n");
- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != 250) {
- fputs($fp, "MAIL FROM: <".preg_replace("/.*\<(.+?)\>.*/", "\\1", $email_from).">\r\n");
- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != 250) {
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) MAIL FROM - $lastmessage", 0);
- return false;
- }
- }
- fputs($fp, "RCPT TO: <".preg_replace("/.*\<(.+?)\>.*/", "\\1", $toemail).">\r\n");
- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != 250) {
- fputs($fp, "RCPT TO: <".preg_replace("/.*\<(.+?)\>.*/", "\\1", $toemail).">\r\n");
- $lastmessage = fgets($fp, 512);
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) RCPT TO - $lastmessage", 0);
- return false;
- }
- fputs($fp, "DATA\r\n");
- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != 354) {
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) DATA - $lastmessage", 0);
- return false;
- }
- $headers .= 'Message-ID: <'.gmdate('YmdHs').'.'.substr(md5($email_message.microtime()), 0, 6).rand(100000, 999999).'@'.$_SERVER['HTTP_HOST'].">{$maildelimiter}";
- fputs($fp, "Date: ".gmdate('r')."\r\n");
- fputs($fp, "To: ".$email_to."\r\n");
- fputs($fp, "Subject: ".$email_subject."\r\n");
- fputs($fp, $headers."\r\n");
- fputs($fp, "\r\n\r\n");
- fputs($fp, "$email_message\r\n.\r\n");
- $lastmessage = fgets($fp, 512);
- if(substr($lastmessage, 0, 3) != 250) {
- runlog('SMTP', "({$_G[setting][mail][server]}:{$_G[setting][mail][port]}) END - $lastmessage", 0);
- }
- fputs($fp, "QUIT\r\n");
复制代码
|