当我们使用PHP的第三方库或工具类进行邮件发送的时候,是否想过一个问题:
为什么我们不能自己写php代码实现邮件发现,而要用别人的库呢?php发送邮件到底是如何实现的?
首先我们要了解发送邮件的基本原理,本文基于SMTP协议实现邮件发送
SMTP(Simple Mail Transfer Protocol)即简单邮件传输协议。简单来说它定义了一组规则,我们只需要依照这个规则来告诉SMTP服务器,我们要发送邮件的发送人,接收人,内容,主题等信息。
然后SMTP服务器依照这组规则来解析我们发送的信息,最后进行邮件发送。
像163,qq等邮件服务器都有提供SMTP服务,我们只要连接上他们的SMTP服务器,然后write数据,就能实现邮件发送了。
其实我们可以不写代码,直接借用Linux的telnet工具来连接smtp服务,进行邮件发送。借此来了解邮件发送的整个流程。
telnet进行邮件发送
我们可以在linux环境下,使用telnet命令,连接163的smtp服务,25端口(一般smtp都是用25端口),借此来理解smtp的传输流程。
telnet smtp.163.com 25
然后会得到以下结果,说明我们连接成功了
Trying 220.181.12.16... Connected to smtp.163.com. Escape character is '^]'. 220 163.com Anti-spam GT for Coremail System (163com[20141201])
接着我们执行以下命令,告诉对方我们的身份标识来自哪里
HELO smtp.163.com
对方会返回给我们一个250 OK
再执行AUTH LOGIN告诉对方我们要开始进行身份认证,然后对方会回应我们一些消息。
后面我们会再输入我们的用户名,密码,发送邮件的内容,发送人,接受人等信息,然后结束对话,smtp服务器就会帮我们把邮件发送出去。
由于smtp协议对邮件内容格式有严格的要求,在命令行中不好执行,所以这里没有将整个过程执行完毕,后面会使用php代码完整实现。
从上面使用telnet连接smtp邮件的过程可以看出来,发送邮件的过程其实很简单,就是连接smtp服务的25端口,依照协议告诉对方我们要发什么邮件即可。这与平台,与编程语言无关。
无论我们用C语言,还是Java或者PHP,只要使用Socket连接SMTP服务器,就能实现邮件发送。
SMTP指令
上面我们使用telnet连接smtp服务时,输入了一些HELO ,AUTH LOGIN等,大家可能会有疑问这些是什么。
其实很简单,这些就是SMTP协议定义的指令,或者说规则,smtp服务器就是通过这些指令才知道我们是想干啥。
常用指令如下:
指令
作用
HELO
向对方邮件服务器发出的标识自己的身份的命令
AUTH LOGIN
即将进行身份认证
MAIL FROM
告诉对方本次邮件发送人是谁
RCPT TO
发送给谁
DATA
告诉对方本次邮件,接下来我们发送邮件具体内容了
QUIT
邮件内容输入完毕后,执行该指令退出
php实现邮件发送
直接上代码
class Mailer { private $host; private $port = 25; private $user; private $pass; private $debug = false; private $sock; public function __construct($host,$port,$user,$pass,$debug = false) { $this->host = $host; $this->port = $port; $this->user = base64_encode($user); //用户名密码一定要使用base64编码才行 $this->pass = base64_encode($pass); $this->debug = $debug; //socket连接 $this->sock = fsockopen($this->host,$this->port); if(!$this->sock){ exit('出错啦'); } //读取smtp服务返回给我们的数据 $response = fgets($this->sock); $this->debug($response); //如果响应中有220返回码,说明我们连接成功了 if(strstr($response,'220') === false){ exit('出错啦'); } } //发送SMTP指令,不同指令的返回码可能不同 public function execCommand($cmd,$return_code){ fwrite($this->sock,$cmd); $response = fgets($this->sock); //输出调试信息 $this->debug('cmd:'.$cmd .';response:'.$response); if(strstr($response,$return_code) === false){ return false; } return true; } public function sendMail($from,$to,$subject,$body){ //detail是邮件的内容,一定要严格按照下面的格式,这是协议规定的 $detail = 'From:'.$from."\r\n"; $detail .= 'To:'.$to."\r\n"; $detail .= 'Subject:'.$subject."\r\n"; $detail .= 'Content-Type: Text/html;'."\r\n"; $detail .= 'charset=gb2312'."\r\n\r\n"; $detail .= $body; $this->execCommand("HELO ".$this->host."\r\n",250); $this->execCommand("AUTH LOGIN\r\n",334); $this->execCommand($this->user."\r\n",334); $this->execCommand($this->pass."\r\n",235); $this->execCommand("MAIL FROM:<".$from.">\r\n",250); $this->execCommand("RCPT TO:<".$to.">\r\n",250); $this->execCommand("DATA\r\n",354); $this->execCommand($detail."\r\n.\r\n",250); $this->execCommand("QUIT\r\n",221); } public function debug($message){ if($this->debug){ echo '<p>Debug:'.$message . PHP_EOL .'</p>'; } } public function __destruct() { fclose($this->sock); } }
调用示例