SMTP协议实现邮件发送

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的传输流程。

  1. telnet smtp.163.com 25

然后会得到以下结果,说明我们连接成功了

  1. Trying 220.181.12.16...
  2. Connected to smtp.163.com.
  3. Escape character is '^]'.
  4. 220 163.com Anti-spam GT for Coremail System (163com[20141201])

接着我们执行以下命令,告诉对方我们的身份标识来自哪里

  1. 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实现邮件发送

  1. class Mailer
  2. {
  3. private $host;
  4. private $port = 25;
  5. private $user;
  6. private $pass;
  7. private $debug = false;
  8. private $sock;
  9. public function __construct($host,$port,$user,$pass,$debug = false)
  10. {
  11. $this->host = $host;
  12. $this->port = $port;
  13. $this->user = base64_encode($user); //用户名密码一定要使用base64编码才行
  14. $this->pass = base64_encode($pass);
  15. $this->debug = $debug;
  16. //socket连接
  17. $this->sock = fsockopen($this->host,$this->port);
  18. if(!$this->sock){
  19. exit('出错啦');
  20. }
  21. //读取smtp服务返回给我们的数据
  22. $response = fgets($this->sock);
  23. $this->debug($response);
  24. //如果响应中有220返回码,说明我们连接成功了
  25. if(strstr($response,'220') === false){
  26. exit('出错啦');
  27. }
  28. }
  29. //发送SMTP指令,不同指令的返回码可能不同
  30. public function execCommand($cmd,$return_code){
  31. fwrite($this->sock,$cmd);
  32. $response = fgets($this->sock);
  33. //输出调试信息
  34. $this->debug('cmd:'.$cmd .';response:'.$response);
  35. if(strstr($response,$return_code) === false){
  36. return false;
  37. }
  38. return true;
  39. }
  40. public function sendMail($from,$to,$subject,$body){
  41. //detail是邮件的内容,一定要严格按照下面的格式,这是协议规定的
  42. $detail = 'From:'.$from."\r\n";
  43. $detail .= 'To:'.$to."\r\n";
  44. $detail .= 'Subject:'.$subject."\r\n";
  45. $detail .= 'Content-Type: Text/html;'."\r\n";
  46. $detail .= 'charset=gb2312'."\r\n\r\n";
  47. $detail .= $body;
  48. $this->execCommand("HELO ".$this->host."\r\n",250);
  49. $this->execCommand("AUTH LOGIN\r\n",334);
  50. $this->execCommand($this->user."\r\n",334);
  51. $this->execCommand($this->pass."\r\n",235);
  52. $this->execCommand("MAIL FROM:<".$from.">\r\n",250);
  53. $this->execCommand("RCPT TO:<".$to.">\r\n",250);
  54. $this->execCommand("DATA\r\n",354);
  55. $this->execCommand($detail."\r\n.\r\n",250);
  56. $this->execCommand("QUIT\r\n",221);
  57. }
  58. public function debug($message){
  59. if($this->debug){
  60. echo '<p>Debug:'.$message . PHP_EOL .'</p>';
  61. }
  62. }
  63. public function __destruct()
  64. {
  65. fclose($this->sock);
  66. }
  67. }

调用示例

  1. $port = 25;
  2. $user = 'username'; //请替换成你自己的smtp用户名
  3. $pass = 'pass'; //请替换成你自己的smtp密码
  4. $host = 'smtp.163.com';
  5. $from = 'xxxxx@163.com';
  6. $to = 'xxxx@qq.com';
  7. $body = 'hello world';
  8. $subjet = '我是标题';
  9. $mailer = new Mailer($host,$port,$user,$pass,true);
  10. $mailer->sendMail($from,$to,$subjet,$body);

在执行指令时有输出调试信息,输出了我们每次执行的指令以及smtp服务返回给我们的响应数据。

因此我们可以看到以下结果

  1. Debug:cmd:HELO smtp.163.com ;response:250 OK
  2. Debug:cmd:AUTH LOGIN ;response:334 dXNlcm5hbWU6
  3. Debug:cmd:aXR6aG91anVuYmxvZ0AxNjMuY29t ;response:334 UGFzc3dvcmQ6
  4. Debug:cmd:QzBjSGRRNe32xiNGFYUE5oag== ;response:235 Authentication successful
  5. Debug:cmd:MAIL FROM: ;response:250 Mail OK
  6. Debug:cmd:RCPT TO:<380472723@qq.com> ;response:250 Mail OK
  7. Debug:cmd:DATA ;response:354 End data with .
  8. 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
  9. Debug:cmd:QUIT ;response:221 Bye

总结

邮件发送步骤

使用socket连接smtp服务
使用smtp指令进行对话,输入身份信息,邮件信息等
结束对话