上一篇文章写了如何通过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邮箱发送一封带有附件的邮件
通过程序接收邮件
查看附件保存时否正确
核心代码
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发送电子邮件文章的内容,一个简单的邮件客户端的基本收信发信功能就有了,但是想实现一个功能完善用户体验好的邮件客户端就需要不断揣摩它,设计它,完善它,希望这篇文章能带给你启发,可以实现一个属于你自己的邮件客户端。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
更新动态
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]