Node接收电子邮件的实例代码


Posted in Javascript onJuly 21, 2017

上一篇文章写了如何通过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接收邮件的过程可以用下面一张图表示

Node接收电子邮件的实例代码

 imap接收邮件测试过程

本次测试的大概过程如下

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

安装node第三方包

npm install --save imap mailparser

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

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

Node接收电子邮件的实例代码

通过程序接收邮件

Node接收电子邮件的实例代码

查看附件保存时否正确

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发送电子邮件文章的内容,一个简单的邮件客户端的基本收信发信功能就有了,但是想实现一个功能完善用户体验好的邮件客户端就需要不断揣摩它,设计它,完善它,希望这篇文章能带给你启发,可以实现一个属于你自己的邮件客户端。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
combox改进版 页面原型参考dojo的,比网上jQuery的那些combox功能强,代码更小
Apr 15 Javascript
js中的this关键字详解
Sep 25 Javascript
js格式化时间和js格式化时间戳示例
Feb 10 Javascript
js Dialog 去掉右上角的X关闭功能
Apr 23 Javascript
jQuery实现表格行上移下移和置顶的方法
May 22 Javascript
JavaScript基础篇(6)之函数表达式闭包
Dec 11 Javascript
JS对象是否拥有某属性如何判断
Feb 03 Javascript
JQuery中Ajax的操作完整例子
Mar 07 Javascript
bootstrap-table.js扩展分页工具栏(增加跳转到xx页)功能
Dec 28 Javascript
快速解决vue-cli不能初始化webpack模板的问题
Mar 20 Javascript
微信小程序实现tab左右切换效果
Nov 15 Javascript
浅谈在vue-cli3项目中解决动态引入图片img404的问题
Aug 04 Javascript
基于JavaScript实现无限加载瀑布流
Jul 21 #Javascript
原生JS实现自定义滚动条效果
Oct 27 #Javascript
JS HTML图片显示Canvas 压缩功能
Jul 21 #Javascript
JavaScript 用fetch 实现异步下载文件功能
Jul 21 #Javascript
详解vue 模版组件的三种用法
Jul 21 #Javascript
Vue单页式应用(Hash模式下)实现微信分享的实例
Jul 21 #Javascript
vue-resource调用promise取数据方式详解
Jul 21 #Javascript
You might like
php利用新浪接口查询ip获取地理位置示例
2014/01/20 PHP
kohana框架上传文件验证规则写法示例
2014/07/14 PHP
thinkphp配置文件路径的实现方法
2016/08/30 PHP
php中的异常和错误浅析
2017/05/03 PHP
PHP根据key删除数组中指定的元素
2019/02/28 PHP
禁止刷新,回退的JS
2006/11/25 Javascript
JavaScript 事件属性绑定带参数的函数
2009/03/13 Javascript
JQuery 选择器 xpath 语法应用
2010/05/13 Javascript
jQueryPad 实用的jQuery测试工具(支持IE,chrome,FF)
2010/05/22 Javascript
使用JS CSS去除IE链接虚线框的三种方法
2013/11/14 Javascript
使用javascript控制cookie显示和隐藏背景图
2014/02/12 Javascript
js 日期比较相关天数代码
2014/04/02 Javascript
JavaScript设置表单上传时文件个数的方法
2015/08/11 Javascript
js判断某个字符出现的次数的简单实例
2016/06/03 Javascript
AngularJS  $on、$emit和$broadcast的使用
2016/09/05 Javascript
Windows下使用Nodejs运行js的方法
2017/09/02 NodeJs
js 实现在2d平面上画8的方法
2018/10/10 Javascript
jQuery实现可以计算进制转换的计算器
2020/10/19 jQuery
vue3.0实现点击切换验证码(组件)及校验
2020/11/18 Vue.js
[05:29]2014DOTA2国际邀请赛 赛后专访:LGDNewbee顺利过关
2014/07/13 DOTA
[37:21]完美世界DOTA2联赛PWL S2 Inki vs Magma 第二场 11.22
2020/11/24 DOTA
python中使用sys模板和logging模块获取行号和函数名的方法
2014/04/15 Python
python 创建弹出式菜单的实现代码
2017/07/11 Python
python中文乱码不着急,先看懂字节和字符
2017/12/20 Python
python高斯分布概率密度函数的使用详解
2019/07/10 Python
python使用tomorrow实现多线程的例子
2019/07/20 Python
安装Anaconda3及使用Jupyter的方法
2020/10/27 Python
Python爬虫之Selenium中frame/iframe表单嵌套页面
2020/12/04 Python
计算机专业大学生的自我评价
2013/11/14 职场文书
大一期末自我鉴定
2013/12/13 职场文书
2014新年寄语
2014/01/20 职场文书
打架检讨书500字
2014/01/29 职场文书
国际经济贸易专业自荐信
2014/06/13 职场文书
入党积极分子学习党的纲领思想汇报
2014/09/13 职场文书
村党支部群众路线教育实践活动对照检查材料
2014/09/26 职场文书
八年级英语教学计划
2015/01/23 职场文书