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 相关文章推荐
JS继承--原型链继承和类式继承
Apr 08 Javascript
JQuery获取样式中的background-color颜色值的问题
Aug 20 Javascript
JS轮播图中缓动函数的封装
Nov 25 Javascript
BootStrap框架中的data-[ ]自定义属性理解(推荐)
Feb 14 Javascript
基于React实现表单数据的添加和删除详解
Mar 14 Javascript
详解vue 模拟后台数据(加载本地json文件)调试
Aug 25 Javascript
jQuery实现的文字逐行向上间歇滚动效果示例
Sep 06 jQuery
详解vue+webpack+express中间件接口使用
Jul 17 Javascript
基于js Canvas实现二次贝塞尔曲线
Dec 25 Javascript
详解Vue.js v-for不支持IE9的解决方法
Dec 29 Javascript
深入学习js函数的隐式参数 arguments 和 this
Jun 24 Javascript
jQuery单页面文字搜索插件jquery.fullsearch.js的使用方法
Feb 04 jQuery
基于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
支付宝接口开发集成支付环境小结
2015/03/17 PHP
[原创]php逐行读取txt文件写入数组的方法
2015/07/02 PHP
Joomla调用系统自带编辑器的实现方法
2016/05/05 PHP
php metaphone()函数及php localeconv() 函数实例解析
2016/05/15 PHP
Kindeditor编辑器添加图片上传水印功能(php代码)
2017/08/03 PHP
php大小写转换函数(strtolower、strtoupper)用法介绍
2017/11/17 PHP
php图片合成方法(多张图片合成一张)
2017/11/25 PHP
一个用js实现的页内搜索代码
2007/05/23 Javascript
jquery 可拖拽的窗体控件实现代码
2010/03/21 Javascript
jQuery中add实现同时选择两个id对象
2010/10/22 Javascript
new Date()问题在ie8下面的处理方法
2014/07/31 Javascript
JS实现简单的tab切换选项卡效果
2016/09/21 Javascript
vue2.0+webpack环境的构造过程
2016/11/08 Javascript
jQuery Validate让普通按钮触发表单验证的方法
2016/12/15 Javascript
npm国内镜像 安装失败的几种解决方案
2017/06/04 Javascript
jquery+css实现侧边导航栏效果
2017/06/12 jQuery
jquery.rotate.js实现可选抽奖次数和中奖内容的转盘抽奖代码
2017/08/23 jQuery
完美解决axios跨域请求出错的问题
2018/02/05 Javascript
微信小程序 swiper 组件遇到的问题及解决方法
2019/05/26 Javascript
微信浏览器下拉黑边解决方案 wScroollFix
2020/01/21 Javascript
jQuery实现移动端笔触canvas电子签名
2020/05/21 jQuery
解决vant的Toast组件时提示not defined的问题
2020/11/11 Javascript
python用Pygal如何生成漂亮的SVG图像详解
2017/02/10 Python
Python爬虫实战:分析《战狼2》豆瓣影评
2018/03/26 Python
Django项目中model的数据处理以及页面交互方法
2018/05/30 Python
自定义Django Form中choicefield下拉菜单选取数据库内容实例
2020/03/13 Python
Python实现列表中非负数保留,负数转化为指定的数值方式
2020/06/04 Python
SQL面试题
2013/04/30 面试题
星空联盟C# .net笔试题
2014/12/05 面试题
社区志愿者心得体会
2014/01/03 职场文书
应届毕业生应聘自荐信范文
2014/02/26 职场文书
鉴定评语大全
2014/05/05 职场文书
阅兵口号
2014/06/19 职场文书
2014党委书记四风对照检查材料思想汇报
2014/09/21 职场文书
领导视察通讯稿
2015/07/18 职场文书
JAVA长虹键法之建造者Builder模式实现
2022/04/10 Java/Android