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 相关文章推荐
JQuery each()函数如何优化循环DOM结构的性能
Dec 10 Javascript
如何在JavaScript中实现私有属性的写类方式(二)
Dec 04 Javascript
JavaScript中的单引号和双引号报错的解决方法
Sep 01 Javascript
js 通过cookie实现刷新不变化树形菜单
Oct 30 Javascript
jQuery使用$.each遍历json数组的简单实现方法
Apr 18 Javascript
Javascript 正则表达式校验数字的简单实例
Nov 02 Javascript
详解vue-router和vue-cli以及组件之间的传值
Jul 04 Javascript
详解Vue取消eslint语法限制
Aug 04 Javascript
详解Vue CLI3 多页应用实践和源码设计
Aug 30 Javascript
layui的面包屑或者表单不显示的解决方法
Sep 05 Javascript
vue resource发送请求的几种方式
Sep 30 Javascript
vue 实现 rem 布局或vw 布局的方法
Nov 13 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
用来解析.htgroup文件的PHP类
2012/09/05 PHP
PHP+Mysql实现多关键字与多字段生成SQL语句的函数
2014/11/05 PHP
jQuery代码优化之基本事件
2011/11/01 Javascript
JQuery选择器特辑 详细小结
2012/05/14 Javascript
JQuery入门——用映射方式绑定不同事件应用示例
2013/02/05 Javascript
如何设置一定时间内只能发送一次请求
2014/02/28 Javascript
node.js使用require()函数加载模块
2014/11/26 Javascript
js中this的用法实例分析
2015/01/10 Javascript
详细解读JavaScript的跨浏览器事件处理
2015/08/12 Javascript
js图片轮播特效代码分享
2015/09/07 Javascript
js右下角弹出提示框示例代码
2016/01/12 Javascript
jQuery+formdata实现上传进度特效遇到的问题
2016/02/24 Javascript
简单实现JS倒计时效果
2016/12/23 Javascript
jQuery实现可兼容IE6的滚动监听功能
2017/09/20 jQuery
浅谈vue单一组件下动态修改数据时的全部重渲染
2018/03/01 Javascript
js中document.write和document.writeln的区别
2018/03/11 Javascript
vue中手机号,邮箱正则验证以及60s发送验证码的实例
2018/03/16 Javascript
微信 jssdk 签名错误invalid signature的解决方法
2019/01/14 Javascript
JavaScript定时器常见用法实例分析
2019/11/15 Javascript
nodejs nedb 封装库与使用方法示例
2020/02/06 NodeJs
vue实现下载文件流完整前后端代码
2020/11/17 Vue.js
[15:35]教你分分钟做大人:天怒法师
2014/10/30 DOTA
Python实现多线程HTTP下载器示例
2017/02/11 Python
python 实现在Excel末尾增加新行
2018/05/02 Python
详解python实现交叉验证法与留出法
2019/07/11 Python
Python 迭代,for...in遍历,迭代原理与应用示例
2019/10/12 Python
html5时钟实现代码
2010/10/22 HTML / CSS
美国在线咖啡、茶和餐厅供应商:LollicupStore
2018/05/04 全球购物
英国领先的独立时装店:Van Mildert
2019/10/28 全球购物
党的群众路线教育实践活动总结报告
2014/04/28 职场文书
教师批评与自我批评心得体会
2014/10/16 职场文书
2014镇党委书记党建工作汇报材料
2014/11/02 职场文书
初中信息技术教学计划
2015/01/22 职场文书
教师岗位职责范本
2015/04/02 职场文书
浅谈mysql增加索引不生效的几种情况
2021/06/23 MySQL
Java生成读取条形码和二维码的简单示例
2021/07/09 Java/Android