Node接收电子邮件

作者 梦想.家 日期 2017-07-20 字数 2,014
Node接收电子邮件

上一篇文章写了如何通过node发送电子邮件,有发送就会有接收嘛,所以这篇文章来说说关于在node中如何接收电子邮件。

邮件协议

在开始这篇文章之前我们首先了解三个协议smtp(Simple Mail Transfer Protocol)简单邮件传输协议,pop3(Post Office Protocol 3)邮局协议第三版本,imap(Internet Mail Access Protocol)internet消息访问协议。

smtp协议

简单邮件传输协议:是一种基于文本的电子邮件传输协议,用于从源地址到目的地址传输邮件的规范,通过它来控制邮件的中转方式,是因特网中用于在邮件服务器之间交换邮件的协议。SMTP是一个“推”的协议,它不允许根据需要从远程服务器上“拉”来消息。要做到这点,邮件客户端必须使用POP3或IMAP。所以发送邮件的时候我们需要简单的了解下面的这两种协议。

pop3协议

POP3协议允许电子邮件客户端下载服务器上的邮件,但是在客户端的操作(如移动邮件、删除邮件、标记已读等),不会反馈到服务器上,比如通过客户端收取了邮箱中的3封邮件并移动到其他文件夹,邮箱服务器上的这些邮件是没有同时被移动的。也就是说POP3协议实际上是下载了一份邮件的副本到本地邮件客户端,而且对本地邮件副本的操作只会影响本地数据。多个邮件客户端里面的邮件的状态可能会不一致。

imap协议

IMAP(Internet消息访问协议)也是提供面向用户的邮件收取服务。常用的版本是IMAP4。与POP3协议类似允许电子邮件客户端下载服务器上的邮件,不同的是,开启了IMAP后,您在电子邮件客户端收取的邮件仍然保留在服务器上,同时在客户端上的操作都会反馈到服务器上,如:删除邮件,标记已读等,服务器上的邮件也会做相应的动作。换句话说,IMAP把远程文件夹当成本地文件夹来操作,它们之间类似于双向同步。这样的好处是,当你在多个邮件客户端看见的邮件的状态是一致的。本次接收邮件我们也使用此协议来实现。

接收邮件测试过程

接收邮件实际上做的是一个邮件客户端的东西,对于底层的实现在npm上有一个写好的第三的库node-imap(node.js的imap客户端模块)这个模块帮助我们封装了很多的底层操作,但是这个模块返回的数据像附件、消息、邮件头等都是未解码的原始数据,所以还需要对数据进行解码,解码的模块在npm上也找到了一个写好的库Mailparser它是一个node高级电子邮件解析器,能够解析即使非常大的数据(100MB+),而且开销相对比较低。

使用imap接收邮件的过程可以用下面一张图表示

imap接收邮件测试过程

本次测试的大概过程如下

  • 在pc上登录qq邮箱
  • 通过qq邮箱发邮件到gamil(不同邮件服务器之间发邮件过程比较复杂,过程略)
  • 在pc上通过写好的基于imap的程序去拉取gmail的邮件,同时本地的修改(标记邮件,删除邮件)会同步到gmail服务器

安装node第三方包

npm install --save imap mailparser

邮件接收服务器我选择Gmail,发送邮件的服务器使用qq邮箱。

使用qq邮箱发送一封带有附件的邮件

发送一封带附件的电子邮件

通过程序接收邮件

node接收到的电子邮件信息

查看附件保存时否正确

查看附件是否保存成功

核心代码

var Imap = require('imap')
var MailParser = require("mailparser").MailParser
var fs = require("fs")

var imap = new Imap({
    user: 'yourname@gmail.com', //你的邮箱账号
    password: 'yourpassword', //你的邮箱密码
    host: 'imap.gmail.com', //邮箱服务器的主机地址
    port: 993, //邮箱服务器的端口地址
    tls: true, //使用安全传输协议
    tlsOptions: { rejectUnauthorized: false } //禁用对证书有效性的检查
});

function openInbox(cb) {
    imap.openBox('INBOX', true, cb);
}

imap.once('ready', function() {

    openInbox(function(err, box) {

        console.log("打开邮箱")

        if (err) throw err;

        imap.search(['UNSEEN', ['SINCE', 'May 20, 2017']], function(err, results) {//搜寻2017-05-20以后未读的邮件

            if (err) throw err;

            var f = imap.fetch(results, { bodies: '' });//抓取邮件(默认情况下邮件服务器的邮件是未读状态)

            f.on('message', function(msg, seqno) {

                var mailparser = new MailParser();

                msg.on('body', function(stream, info) {

                    stream.pipe(mailparser);//将为解析的数据流pipe到mailparser

                    //邮件头内容
                    mailparser.on("headers", function(headers) {
						console.log("邮件头信息>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>");
                        console.log("邮件主题: " + headers.get('subject'));
                        console.log("发件人: " + headers.get('from').text);
                        console.log("收件人: " + headers.get('to').text);
                    });

                    //邮件内容
					
                    mailparser.on("data", function(data) {
                        if (data.type === 'text') {//邮件正文
							console.log("邮件内容信息>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>");
                            console.log("邮件内容: " + data.html);
                        }
                        if (data.type === 'attachment') {//附件
							console.log("邮件附件信息>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>");
                            console.log("附件名称:"+data.filename);//打印附件的名称
                            data.content.pipe(fs.createWriteStream(data.filename));//保存附件到当前目录下
                            data.release();
                        }
                    });

                });
                msg.once('end', function() {
                    console.log(seqno + '完成');
                });
            });
            f.once('error', function(err) {
                console.log('抓取出现错误: ' + err);
            });
            f.once('end', function() {
                console.log('所有邮件抓取完成!');
                imap.end();
            });
        });
    });
});

imap.once('error', function(err) {
    console.log(err);
});

imap.once('end', function() {
    console.log('关闭邮箱');
});

imap.connect();


默认情况下抓取邮件后邮件服务器的邮件状态为未读,如果要在抓取后让邮箱服务器中的邮件状态变为已读,可以修改

var f = imap.fetch(results, { bodies: '' });

var f = imap.fetch(results, { bodies: '', markSeen: true }); 

参考文档

查看node-imap详细文档和api请点击这里 查看Mailparser详细文档请点击这里

后记

如果google邮箱如果开启了二次认证,那么你需要在google后台生成一个专用密码来登录google邮箱拉取邮件。

通过一个简单的例子实现了如何使用node和imap协议来接收邮件,结合上篇node发送电子邮件文章的内容,一个简单的邮件客户端的基本收信发信功能就有了,但是想实现一个功能完善用户体验好的邮件客户端就需要不断揣摩它,设计它,完善它,希望这篇文章能带给你启发,可以实现一个属于你自己的邮件客户端。




本文完

本文如有误,请不吝赐教!

原文标题:Node接收电子邮件

原文链接:http://www.mengxiangjia.info/2017/07/20/node-receive-mail/

版权声明:自由转载-非商用-非衍生-保持署名 | Creative Commons BY-NC-ND 3.0