SMTP协议
当我们使用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的传输流程。
然后会得到以下结果,说明我们连接成功了
接着我们执行以下命令,告诉对方我们的身份标识来自哪里
对方会返回给我们一个250 OK
再执行AUTH LOGIN告诉对方我们要开始进行身份认证,然后对方会回应我们一些消息。
后面我们会再输入我们的用户名,密码,发送邮件的内容,发送人,接受人等信息,然后结束对话,smtp服务器就会帮我们把邮件发送出去。
由于smtp协议对邮件内容格式有严格的要求,在命令行中不好执行,所以这里没有将整个过程执行完毕,后面会使用PHP代码完整实现。
从上面使用telnet连接smtp邮件的过程可以看出来,发送邮件的过程其实很简单,就是连接smtp服务的25端口,依照协议告诉对方我们要发什么邮件即可。这与平台,与编程语言无关。
无论我们用C语言,还是Java或者PHP,只要使用Socket连接SMTP服务器,就能实现邮件发送。
SMTP指令
上面我们使用telnet连接smtp服务时,输入了一些HELO ,AUTH LOGIN等,大家可能会有疑问这些是什么。
其实很简单,这些就是SMTP协议定义的指令,或者说规则,smtp服务器就是通过这些指令才知道我们是想干啥。
常用指令如下:
Box-sizing: border-Box; border-top: rgb(238,238) 1px solid; border-right: rgb(238,238) 1px solid; vertical-align: top; border-bottom: rgb(238,238) 1px solid; padding-bottom: 8px; padding-top: 8px; padding-left: 8px; border-left: rgb(238,238) 1px solid; line-height: 20px; padding-right: 8px">指令 | 作用 |
---|---|
PHP实现邮件发送
直接上代码
{
$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 '
Debug:'.$message . PHP_EOL .'
';}
}
public function __destruct()
{
fclose($this->sock);
}
}
调用示例
在执行指令时有输出调试信息,输出了我们每次执行的指令以及smtp服务返回给我们的响应数据。
因此我们可以看到以下结果
Debug:cmd:AUTH LOGIN ;response:334 dXNlcm5hbWU6
Debug:cmd:aXR6aG91anVuYmxvZ0AxNjMuY29t ;response:334 UGFzc3dvcmQ6
Debug:cmd:QzBjSGRRNe32xiNGFYUE5oag== ;response:235 Authentication successful
Debug:cmd:MAIL FROM: ;response:250 Mail OK
Debug:cmd:RCPT TO:380472723@qq.com ;response:250 Mail OK
Debug:cmd:DATA ;response:354 End data with .
Debug:cmd:From:itzhoujunblog@163.com To:380472723@qq.com Subject:我是标题 Content-Type: Text/html; charset=gb2312 hello world . ;response:250 Mail OK queued as smtp11,D8CowACXHE5APdNYCo0hAQ--.19144S2 1490238785
Debug:cmd:QUIT ;response:221 Bye
总结
邮件发送步骤
- 使用socket连接smtp服务
- 使用smtp指令进行对话,输入身份信息,邮件信息等
- 结束对话