使用Node解析EML文件

什么是EML文件

通过的说,电子邮件导出后的文件格式就是.eml文件,比如使用outlook163邮箱等等电子邮件程序将电子邮件导出后,就可以得到.eml文件,EML文件应该符合RFC 5322规范,这样EML文件就可以在不同的邮件客户端之间流通。

也就是说,使用163邮箱客户端导出的eml文件,完全可以在Ooutlook中『基本一致』的打开

之所以说基本一致,因为还是有很多兼容性问题要处理,而且很多邮箱客户端还有独有的私有协议内容,对应的样式在其他邮箱客户端只能以基本的样式进行展示

以163邮箱为例,如下图,将邮件导出,就得到了预约直播,听专家分享《面向大规模数据的云端管理,百度沧海·存储产品解析》.eml文件
在这里插入图片描述
使用邮箱大师,就可以将这个文件导入,并进行展示:
在这里插入图片描述
展示与原邮件基本是一致的
在这里插入图片描述

实现EML导入

在我们的灵犀办公客户端上,也要实现这个导入EML文件的功能,这个功能有现成的库来支持,可选择的有三个库:

三个工具使用的解析的结果大同小异,我们选用的是MailParser(额外说一句,NodeMailer功能非常强大,可以用它构建一个自己的邮件收发系统)

按照文档很容易就能实现一个简单的EML解析器:

const fse = require('fs-extra');
const simpleParser = require('mailparser').simpleParser;

function parseEml(filePath) {
   
  return fse.stat(filePath).then(() => {
   
    return fse.readFile(filePath).then((file) => {
   
      return simpleParser(file);
    });
  });
}

解析的结果包含了下面的字段:
在这里插入图片描述
在我们的系统中,根据这些属性,可以构造出一个完整的MailEntitiy结构,用于完成常规的服务端返回的邮件数据解析流程。到此解析就流程就完了。

就这?

当然不是,为了提高解析性能,我们还在客户端本地的数据库,使用文件地址和文件名称组合为key,在本地数据库记录了结果。当然这都是小优化,直到功能上线后,QA报了一个线上问题,有一封邮件,我们的客户端打开后,样式乱了,结果如下:
在这里插入图片描述
导入导服务端之后却是正常的
在这里插入图片描述
用邮箱大师也是正常的,那我们出了什么问题呢?

用MailParser解析之后,发现解析的结果里面,attachments是有值的,而htmlfalsetext里面就是我们的客户端解析后的数据,并没有相应的内联样式:
在这里插入图片描述
一开始以为是参数的问题,一通调试后发现然并卵,后来以为是编码的问题,因为这封邮件的charsetgb2312,结果发现也没有用(虽然后面编码还是要处理的)

咨询了服务端的同学,他给出的答案一针见血,邮件的附件里面是不是有一个dat附件,需要对dat附件进行二次解析才可以

winmail.dat

winmai.dat是Microsoft Outlook是特殊格式附件,如果Outlook寄出富文本格式的信件,其他邮件客户端收到的时候,就会产生这个文件。真正的内容和样式实际上都是在这个附件中。
在这里插入图片描述

所以需要对这个文件进行二次解析,在NPM上找到了node-tnef这个工具,使用它就可以解析winmai.dat附件。由于我们在解析eml文件时,附件已经是Buffer格式的数据了,所以直接使用tnef.parseBuffer这个方法就可以,得到的content里面包含了BodyHTML数据,把这个带有样式的富文本数据格式赋值给eml解析后的html字段(原本是false)就可以解析出正确的数据了

function handleDatAttachments(parsedMail, encoding) {
   
  const datAttach = parsedMail.attachments.find(
    (v) => v.contentType === 'application/ms-tnef' && v.filename && v.filename.endsWith('dat')
  );
  if (datAttach) {
   
    return new Promise((resolve) 
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Rust 中解析 .eml(电子邮件文件通常涉及到读取二进制数据并使用库来解析电子邮件的结构。Rust 有一个名为 `EmailParser` 的库,它可以帮助你处理这个任务。以下是一个简单的步骤指南: 1. **添加依赖库**: 首先,你需要在你的 `Cargo.toml` 文件中添加 `email-lexer` 和 `email-parser` 作为依赖。例如: ```toml [dependencies] email-lexer = "0.16" email-parser = "0.16" ``` 2. **导入和使用库**: 在你的 Rust 代码中,引入这些库并创建一个 `EmailParser` 的实例: ```rust extern crate email; extern crate email_lexical; use std::fs::File; use std::io::{BufReader, BufRead}; use email::parser::Parser; use email_lexical::reader::Reader; ``` 3. **读取和解析文件**: 打开并读取 .eml 文件的内容,然后用 `EmailParser` 解析: ```rust async fn parse_email(file_path: &str) -> Result<email::Message, Box<dyn std::error::Error>> { let file = File::open(file_path)?; let reader = BufReader::new(file); let mut email_reader = Reader::new(reader); Parser::parse_message(email_reader) .await .map(|msg| msg.unwrap()) .map_err(|err| Box::new(err)) // 将错误转换为 `Box<dyn std::error::Error>` } ``` 4. **处理解析结果**: 获取到 `email::Message` 对象后,你可以遍历邮件头和正文,提取所需的信息: ```rust async fn main() { let file_path = "path/to/your/email.eml"; let message = parse_email(&file_path).await?; // ... 进行邮件内容的处理,如打印头信息或正文 println!("Subject: {}", message.headers().get("Subject").unwrap_or_default()); println!("Body: {}", message.body_text().unwrap_or_default()); } ``` **相关问题--:** 1. 如何处理解析 .eml 文件时可能出现的错误? 2. `email::Message` 结构体提供了哪些常用的方法来访问邮件内容? 3. Rust 中如何处理异步 I/O,以便在解析文件时避免阻塞?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值