写在前面

电子邮件是啥就不用介绍了吧,利用程序发送邮件,可以实现:客户财务报表推送、服务异常预警、自动订阅文章等等功能。

php来发送邮件的类库网上也有。比如:PHPMailer 等

但是由于类库年代久远,或者自己使用过程中出现了一些异常错误,导致一系列苦恼。

所以写下这篇文章,来讲明SMTP邮件服务器的原理,让你在调试对接的过程中,有思路可循。

基础知识储备

TCP:TCP是一种面向连接的、可靠的、基于字节流的传输层通信协议。

先去看看TCP协议的基础概念:比如连接握手三次、断开、收发包等等流程。

比如我们访问一个网站,使用的是HTTP协议,HTTP协议是基于TCP协议的。

我们本次要讲的SMTP也是基于TCP协议的。

SSL:加密传输

比如我们的http网站和https网站,在传输过程中加密,会比较安全。

大部分的SMTP服务器也会要求加密传输内容。

SMTP协议的定义

简单邮件传输协议 (Simple Mail Transfer Protocol 简称 SMTP)

是一个相对简单的基于文本的协议。

在发送方(客户端)和接收方(服务器)间创建TCP连接之后

那么接下来就是一个合法的SMTP会话了。(SMTP会话的本质只是一个普通TCP,只是会话的消息按照规范组装发送)

在下面的对话中,所有客户端发送的都以C:作为前缀,所有服务器发送的都以S:作为前缀。

S: 220 smtp.qq.com ESMTP Postfix
C: HELO smtp.other.com
S: 250 Hello smtp.other.com
C: MAIL FROM: <59419979@qq.com>
S: 250 Ok
C: RCPT TO: <123456@qq.com>
S: 250 Ok
C: DATA
S: 354 End data with <CR><LF>.<CR><LF>
C: Subject: 邮件名
C: From: Siam<59419979@qq.com>
C: To: 超级牛逼的QQ号<123456@qq.com>
C:
C: Hello,
C: 这是一个测试邮件.
C: Goodbye~
C: .
S: 250 Ok: queued as 12345
C: quit
S: 221 Bye

这就是发送邮件的一个简单的会话过程,其实基本上是一问一答:
① 服务端:连接上了 由服务器推送给客户端 220状态码 连接成功 这里是QQ的邮件服务器
② 客户端:你好 我是网易的邮件服务器(或者其他…)
③ 服务端:哦好的 网易邮件服务器
④ 客户端:我是59419979账号,我要发送给123456
⑤ 服务端:好的、
⑥ 客户端:我要写内容了。
⑦ 服务端:好的,最后以 . 换行 结束哦!
⑧ 客户端:写写写写
⑨ 客户端:我写完了
⑩ 服务端:好的 接收到记录了
…然后就是退出逻辑了

当然,这其中还可以有其他逻辑和报错提示等

比如要授权登陆(QQ号不能随便让别人操作吧?)
比如要求会话是SSL加密传输的,明文传输我不接受,断开连接了哦。(下文演示会出现这个情况)

实战本地连接SMTP

下载一个简单的TCP测试软件,打开,创建连接。

QQ的SMTP服务器地址为:smtp.qq.com 端口为 465 或者 587

然后点击连接


因为到这里,本地测试的工具不支持加密传输,所以运行不了了。

接着是使用swoole提供的tcp客户端来链接操作。

以下演示代码仅提供核心部分。

$this->client = new Client( SWOOLE_TCP | SWOOLE_SSL);
$this->client->set([
    'open_eof_check' => true,
    'package_eof' => "\r\n",
    'package_max_length' => 1024 * 1024 * 2,
]);
if ($this->client->connect('smtp.qq.com', 465,$this->timeout) === false) {
     throw new Exception("connect fail");
 }
$str = $this->recvCodeCheck('220');
$ehloHost = explode(' ',$str)[1];
$this->client->send("ehlo {$ehloHost}\r\n");
//先看是否得到250应答,并清除多余应答
$this->recvCodeCheck('250');
while (1){
    $peek = $this->client->recv($this->timeout);
    if(empty($peek)){
        throw new Exception('waiting 250 code error');
    }else{
        if(substr($peek,3,1) != '-'){
            break;
        }
    }
}
// 验证身份 登陆 不是所有的邮箱都一样的流程
$this->client->send("auth login\r\n");
$this->recvCodeCheck('334');
$this->client->send(base64_encode('59419979')."\r\n");
$this->recvCodeCheck('334');
$this->client->send(base64_encode('这里是授权码 在QQ邮箱后台拿到')."\r\n");
$this->recvCodeCheck('235');
//start send data
$this->client->send("mail from:<59419979@qq.com>\r\n");
$this->recvCodeCheck('250');
$this->client->send("rcpt to:<123456@qq.com>\r\n");
$this->recvCodeCheck('250');
$this->client->send("data\r\n");
$this->recvCodeCheck('354');
//build body
$mail = "";
$mail.= "From: Siam<59419979@qq.com>\r\n";
$mail.= "To: 不认识<123456@qq.com>\r\n";
$mail.= "Subject: Test\r\n";
//构造body
$mail.= "正文\r\n";
$this->client->send($mail);
$this->client->send(".\r\n");
$this->recvCodeCheck('250');
$this->client->send("quit\r\n");
$this->recvCodeCheck('221');

在easyswoole框架中,会提供smtp类库,以上代码就是部分的实现

使用类库可以直接使用