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 相关文章推荐
Javascript 生成指定范围数值随机数
Jan 09 Javascript
js动态为代码着色显示行号
May 29 Javascript
jquery scroll()区分横向纵向滚动条的方法
Apr 04 Javascript
iframe里面的元素触发父窗口元素事件的jquery代码
Oct 19 Javascript
javascript实现选中复选框后相关输入框变灰不可用的方法
Aug 11 Javascript
jquery mobile开发常见问题分析
Jan 21 Javascript
javascript三种代码注释方法
Jun 02 Javascript
如何判断出一个js对象是否一个dom对象
Nov 24 Javascript
Bootstrap 模态框(Modal)插件代码解析
Dec 21 Javascript
微信小程序 页面跳转事件绑定的实例详解
Sep 20 Javascript
layui问题之模拟select点击事件的实例讲解
Aug 15 Javascript
JavaScript手写数组的常用函数总结
Nov 22 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 一个比较完善的简单文件上传
2010/03/25 PHP
PHP中PDO的错误处理
2011/09/04 PHP
php的数组与字符串的转换函数整理汇总
2013/07/18 PHP
PHP 字符串长度判断效率更高的方法
2014/03/02 PHP
PHP中使用Memache作为进程锁的操作类分享
2015/03/30 PHP
php中smarty模板条件判断用法实例
2015/06/11 PHP
yii2使用gridView实现下拉列表筛选数据
2017/04/10 PHP
php+javascript实现的动态显示服务器运行程序进度条功能示例
2017/08/07 PHP
js 加载时自动调整图片大小
2008/05/28 Javascript
在浏览器窗口上添加遮罩层的方法
2012/11/12 Javascript
javascript为按钮注册回车事件(设置默认按钮)的方法
2015/05/09 Javascript
HTML5 实现的一个俄罗斯方块实例代码
2016/09/19 Javascript
AngularJS实现给动态生成的元素绑定事件的方法
2016/12/14 Javascript
jQuery实现遮罩层登录对话框
2016/12/29 Javascript
图片懒加载插件实例分享(含解析)
2017/01/09 Javascript
ES6新特性之数组、Math和扩展操作符用法示例
2017/04/01 Javascript
JavaScript实现计算圆周率到小数点后100位的方法示例
2018/05/08 Javascript
ES6知识点整理之函数对象参数默认值及其解构应用示例
2019/04/17 Javascript
微信小程序和百度的语音识别接口详解
2019/05/06 Javascript
Python中的is和id用法分析
2015/01/26 Python
用Python实现一个简单的线程池
2015/04/07 Python
Python中Iterator迭代器的使用杂谈
2016/06/20 Python
python读取二进制mnist实例详解
2017/05/31 Python
一个月入门Python爬虫学习,轻松爬取大规模数据
2018/01/03 Python
python实现图书借阅系统
2019/02/20 Python
Python代理IP爬虫的新手使用教程
2019/09/05 Python
Pytorch中Tensor与各种图像格式的相互转化详解
2019/12/26 Python
python脚本实现mp4中的音频提取并保存在原目录
2020/02/27 Python
Python爬虫之Selenium实现关闭浏览器
2020/12/04 Python
浅谈html5与APP混合开发遇到的问题总结
2018/03/20 HTML / CSS
学生就业推荐信
2013/11/13 职场文书
劲霸男装广告词
2014/03/21 职场文书
委托书怎么写
2014/07/31 职场文书
2015年度个人教学工作总结
2015/05/20 职场文书
2016年九九重阳节活动总结
2016/04/01 职场文书
Python Pandas模块实现数据的统计分析的方法
2021/06/24 Python