前文写了使用vmime库通过POP3协议收取邮件,继续写对邮件的解析。
解析邮件相对比较简单,需要将收取的邮件,重新从字符串转换成vmime::message格式,然后就可以获取到自己需要部分的内容了。
首先将vmime::string格式转换为vmime::message:
vmime::utility::ref<vmime::message> mail = vmime::create<vmime::message>();
mail->parse(mailContent);
vmime还提供了一个简单的帮助类vmime::messageParser方便对message进行解析。
message主要包含了邮件头和邮件内容,内容又因为multi-part的邮件格式规定,被拆分成了多个vmime::textPart。通常使用到的textPart的子类,有vmime::htmlTextPart和vmime::plainTextPart,分别对应邮件body中的content-type为text/html和text/plain。
代码:
vmime::messageParser mp(mail);
for (int i = 0; i < mp.getTextPartCount(); ++i) //遍历所有的textPart
{
vmime::utility::ref<const vmime::textPart> text = mp.getTextPartAt(i);
if (text->getType().getSubType() == vmime::mediaTypes::TEXT_HTML) //text/html
{
vmime::utility::ref<const vmime::htmlTextPart> htmlText = text.dynamicCast<const vmime::htmlTextPart>();
vmime::utility::outputStreamStringAdapter htmlOut(htmlContent);
vmime::utility::charsetFilteredOutputStream utf8Out(htmlText->getCharset(), vmime::charset("utf-8"), htmlOut); //强制转换正文为utf8编码
htmlText->getText()->extract(utf8Out);
utf8Out.flush();
}
else if (text->getType().getSubType() == vmime::mediaTypes::TEXT_PLAIN) //text/plain
{
vmime::utility::ref<const vmime::plainTextPart> plainText = text.dynamicCast<const vmime::plainTextPart>();
vmime::utility::outputStreamStringAdapter plainOut(plainTextContent);
vmime::utility::charsetFilteredOutputStream utf8Out(plainText->getCharset(), vmime::charset("utf-8"), plainOut);
plainText->getText()->extract(utf8Out);
utf8Out.flush();
}
}
对于html个是的邮件正文,还可以遍历获取里面的embeddedObject,如嵌入的附件图片等,不过目前没有这样的需求,就没有去尝试了。
在真正执行的时候,又发现了一个问题,必须在开始使用前,调用vmime::platform::setHandler<vmime::platforms::posix::posixHandler>();设置平台相关的handler,这里设置的是符合posix的平台,windows貌似也有对应的handler。