2004-10-05 10:28:21 +00:00
|
|
|
//
|
|
|
|
// VMime library (http://vmime.sourceforge.net)
|
|
|
|
// Copyright (C) 2002-2004 Vincent Richard <vincent@vincent-richard.net>
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License as
|
|
|
|
// published by the Free Software Foundation; either version 2 of
|
|
|
|
// the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "messageParser.hpp"
|
|
|
|
|
|
|
|
#include "defaultAttachment.hpp"
|
|
|
|
#include "textPartFactory.hpp"
|
|
|
|
|
|
|
|
|
|
|
|
namespace vmime
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
messageParser::messageParser(const string& buffer)
|
|
|
|
{
|
|
|
|
vmime::message msg;
|
|
|
|
msg.parse(buffer);
|
|
|
|
|
|
|
|
parse(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
messageParser::messageParser(const message& msg)
|
|
|
|
{
|
|
|
|
parse(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
messageParser::~messageParser()
|
|
|
|
{
|
|
|
|
free_container(m_attach);
|
|
|
|
free_container(m_textParts);
|
|
|
|
|
|
|
|
for (std::map <attachment*, contentDispositionField*>::iterator
|
|
|
|
it = m_attachInfo.begin() ; it != m_attachInfo.end() ; ++it)
|
|
|
|
{
|
|
|
|
delete ((*it).second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void messageParser::parse(const message& msg)
|
|
|
|
{
|
|
|
|
// Header fields (if field is present, copy its value, else do nothing)
|
2004-10-21 15:05:47 +00:00
|
|
|
#define TRY_FIELD(var, type, name) \
|
|
|
|
try { var = dynamic_cast<type&>(*msg.getHeader()->findField(name)).getValue(); } \
|
|
|
|
catch (exceptions::no_such_field) { }
|
|
|
|
|
|
|
|
TRY_FIELD(m_from, mailboxField, fields::FROM);
|
|
|
|
|
|
|
|
TRY_FIELD(m_to, addressListField, fields::TO);
|
|
|
|
TRY_FIELD(m_cc, addressListField, fields::CC);
|
|
|
|
TRY_FIELD(m_bcc, addressListField, fields::BCC);
|
2004-10-05 10:28:21 +00:00
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
TRY_FIELD(m_subject, textField, fields::SUBJECT);
|
2004-10-05 10:28:21 +00:00
|
|
|
|
|
|
|
#undef TRY_FIELD
|
|
|
|
|
|
|
|
// Date
|
|
|
|
try
|
|
|
|
{
|
2004-10-21 15:05:47 +00:00
|
|
|
vmime::relayField& recv = dynamic_cast <vmime::relayField&>
|
|
|
|
(*msg.getHeader()->findField(fields::RECEIVED));
|
|
|
|
|
|
|
|
m_date = recv.getValue().getDate();
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
catch (vmime::exceptions::no_such_field&)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2004-10-21 15:05:47 +00:00
|
|
|
vmime::dateField& date = dynamic_cast <vmime::dateField&>
|
|
|
|
(*msg.getHeader()->findField(fields::DATE));
|
|
|
|
|
|
|
|
m_date = date.getValue();
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
catch (vmime::exceptions::no_such_field&)
|
|
|
|
{
|
|
|
|
m_date = datetime::now();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attachments
|
|
|
|
findAttachments(msg);
|
|
|
|
|
|
|
|
// Text parts
|
|
|
|
findTextParts(msg, msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void messageParser::findAttachments(const bodyPart& part)
|
|
|
|
{
|
|
|
|
// We simply search for parts that are not "Content-disposition: inline".
|
2004-10-21 15:05:47 +00:00
|
|
|
for (int i = 0 ; i < part.getBody()->getPartCount() ; ++i)
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
2004-10-21 15:05:47 +00:00
|
|
|
const bodyPart& p = *part.getBody()->getPartAt(i);
|
|
|
|
const header& hdr = *p.getHeader();
|
|
|
|
const body& bdy = *p.getBody();
|
2004-10-05 10:28:21 +00:00
|
|
|
|
|
|
|
// Is this part an attachment?
|
|
|
|
bool isAttachment = false;
|
|
|
|
const contentDispositionField* contentDispField = NULL;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const contentDispositionField& cdf = dynamic_cast<contentDispositionField&>
|
2004-10-21 15:05:47 +00:00
|
|
|
(*hdr.findField(fields::CONTENT_DISPOSITION));
|
2004-10-05 10:28:21 +00:00
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
if (cdf.getValue().getName() != dispositionTypes::INLINE)
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
|
|
|
contentDispField = &cdf;
|
|
|
|
isAttachment = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (exceptions::no_such_field)
|
|
|
|
{
|
|
|
|
// No "Content-disposition" field: assume "attachment" if
|
|
|
|
// type is not "text/..." or "multipart/...".
|
|
|
|
mediaType type;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const contentTypeField& ctf = dynamic_cast<contentTypeField&>
|
2004-10-21 15:05:47 +00:00
|
|
|
(*hdr.findField(fields::CONTENT_TYPE));
|
2004-10-05 10:28:21 +00:00
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
type = ctf.getValue();
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
catch (exceptions::no_such_field)
|
|
|
|
{
|
|
|
|
// No "Content-type" field: assume "application/octet-stream".
|
|
|
|
type = mediaType(mediaTypes::APPLICATION,
|
|
|
|
mediaTypes::APPLICATION_OCTET_STREAM);
|
|
|
|
}
|
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
if (type.getType() != mediaTypes::TEXT &&
|
|
|
|
type.getType() != mediaTypes::MULTIPART)
|
|
|
|
{
|
2004-10-05 10:28:21 +00:00
|
|
|
isAttachment = true;
|
2004-10-21 15:05:47 +00:00
|
|
|
}
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (isAttachment)
|
|
|
|
{
|
|
|
|
// Determine the media type of this attachment
|
|
|
|
mediaType type;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const contentTypeField& ctf = dynamic_cast<contentTypeField&>
|
2004-10-21 15:05:47 +00:00
|
|
|
(*hdr.findField(fields::CONTENT_TYPE));
|
2004-10-05 10:28:21 +00:00
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
type = ctf.getValue();
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
catch (exceptions::no_such_field)
|
|
|
|
{
|
|
|
|
// No "Content-type" field: assume "application/octet-stream".
|
|
|
|
type = mediaType(mediaTypes::APPLICATION,
|
|
|
|
mediaTypes::APPLICATION_OCTET_STREAM);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the description (if available)
|
|
|
|
text description;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const textField& cd = dynamic_cast<textField&>
|
2004-10-21 15:05:47 +00:00
|
|
|
(*hdr.findField(fields::CONTENT_DESCRIPTION));
|
2004-10-05 10:28:21 +00:00
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
description = cd.getValue();
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
catch (exceptions::no_such_field)
|
|
|
|
{
|
|
|
|
// No description available.
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct the attachment object
|
|
|
|
attachment* attach = new defaultAttachment
|
2004-10-21 15:05:47 +00:00
|
|
|
(bdy.getContents(), bdy.getEncoding(), type, description);
|
2004-10-05 10:28:21 +00:00
|
|
|
|
|
|
|
if (contentDispField != NULL)
|
|
|
|
{
|
|
|
|
m_attachInfo.insert(std::map <attachment*, contentDispositionField*>::
|
2004-10-21 15:05:47 +00:00
|
|
|
value_type(attach, dynamic_cast <contentDispositionField*>
|
2004-10-05 10:28:21 +00:00
|
|
|
(contentDispField->clone())));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the attachment to the list
|
|
|
|
m_attach.push_back(attach);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to find attachments in sub-parts
|
2004-10-21 15:05:47 +00:00
|
|
|
if (bdy.getPartCount())
|
|
|
|
findAttachments(p);
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void messageParser::findTextParts(const bodyPart& msg, const bodyPart& part)
|
|
|
|
{
|
|
|
|
// Handle the case in which the message is not multipart: if the body part is
|
|
|
|
// "text/*", take this part.
|
2004-10-21 15:05:47 +00:00
|
|
|
if (part.getBody()->getPartCount() == 0)
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
|
|
|
mediaType type(mediaTypes::TEXT, mediaTypes::TEXT_PLAIN);
|
|
|
|
bool accept = false;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const contentTypeField& ctf = dynamic_cast<contentTypeField&>
|
2004-10-21 15:05:47 +00:00
|
|
|
(*msg.getHeader()->findField(fields::CONTENT_TYPE));
|
2004-10-05 10:28:21 +00:00
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
if (ctf.getValue().getType() == mediaTypes::TEXT)
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
2004-10-21 15:05:47 +00:00
|
|
|
type = ctf.getValue();
|
2004-10-05 10:28:21 +00:00
|
|
|
accept = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (exceptions::no_such_field)
|
|
|
|
{
|
|
|
|
// No "Content-type" field: assume "text/plain".
|
|
|
|
accept = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (accept)
|
|
|
|
{
|
|
|
|
textPart* textPart = textPartFactory::getInstance()->create(type);
|
|
|
|
textPart->parse(msg, msg, msg);
|
|
|
|
|
|
|
|
m_textParts.push_back(textPart);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Multipart message
|
|
|
|
else
|
|
|
|
{
|
|
|
|
findSubTextParts(msg, part);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool messageParser::findSubTextParts(const bodyPart& msg, const bodyPart& part)
|
|
|
|
{
|
|
|
|
// In general, all the text parts are contained in parallel in the same
|
|
|
|
// parent part (or message).
|
|
|
|
// So, wherever the text parts are, all we have to do is to find the first
|
|
|
|
// MIME part which is a text part.
|
|
|
|
|
|
|
|
std::vector <const bodyPart*> textParts;
|
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
for (int i = 0 ; i < part.getBody()->getPartCount() ; ++i)
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
2004-10-21 15:05:47 +00:00
|
|
|
const bodyPart& p = *part.getBody()->getPartAt(i);
|
|
|
|
|
2004-10-05 10:28:21 +00:00
|
|
|
try
|
|
|
|
{
|
|
|
|
const contentTypeField& ctf = dynamic_cast<contentTypeField&>
|
2004-10-21 15:05:47 +00:00
|
|
|
(*p.getHeader()->findField(fields::CONTENT_TYPE));
|
2004-10-05 10:28:21 +00:00
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
if (ctf.getValue().getType() == mediaTypes::TEXT)
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
2004-10-21 15:05:47 +00:00
|
|
|
textParts.push_back(&p);
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (exceptions::no_such_field)
|
|
|
|
{
|
|
|
|
// No "Content-type" field.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (textParts.size())
|
|
|
|
{
|
|
|
|
// Okay. So we have found at least one text part
|
2004-10-21 15:05:47 +00:00
|
|
|
for (std::vector <const bodyPart*>::const_iterator p = textParts.begin() ;
|
|
|
|
p != textParts.end() ; ++p)
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
|
|
|
const contentTypeField& ctf = dynamic_cast<contentTypeField&>
|
2004-10-21 15:05:47 +00:00
|
|
|
(*(*p)->getHeader()->findField(fields::CONTENT_TYPE));
|
2004-10-05 10:28:21 +00:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2004-10-21 15:05:47 +00:00
|
|
|
textPart* textPart = textPartFactory::getInstance()->create(ctf.getValue());
|
2004-10-05 10:28:21 +00:00
|
|
|
textPart->parse(msg, part, **p);
|
|
|
|
|
|
|
|
m_textParts.push_back(textPart);
|
|
|
|
}
|
|
|
|
catch (exceptions::no_factory_available& e)
|
|
|
|
{
|
|
|
|
// Content-type not recognized.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//else
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
for (int i = 0 ; !found && (i < part.getBody()->getPartCount()) ; ++i)
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
2004-10-21 15:05:47 +00:00
|
|
|
found = findSubTextParts(msg, *part.getBody()->getPartAt(i));
|
2004-10-05 10:28:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
const contentDispositionField* messageParser::getAttachmentInfo(const attachment* a) const
|
2004-10-05 10:28:21 +00:00
|
|
|
{
|
|
|
|
std::map <attachment*, contentDispositionField*>::const_iterator
|
2004-10-21 15:05:47 +00:00
|
|
|
it = m_attachInfo.find(const_cast <attachment*>(a));
|
2004-10-05 10:28:21 +00:00
|
|
|
|
|
|
|
return (it != m_attachInfo.end() ? (*it).second : NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-10-21 15:05:47 +00:00
|
|
|
const mailbox& messageParser::getExpeditor() const
|
|
|
|
{
|
|
|
|
return (m_from);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const addressList& messageParser::getRecipients() const
|
|
|
|
{
|
|
|
|
return (m_to);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const addressList& messageParser::getCopyRecipients() const
|
|
|
|
{
|
|
|
|
return (m_cc);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const addressList& messageParser::getBlindCopyRecipients() const
|
|
|
|
{
|
|
|
|
return (m_bcc);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const text& messageParser::getSubject() const
|
|
|
|
{
|
|
|
|
return (m_subject);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const datetime& messageParser::getDate() const
|
|
|
|
{
|
|
|
|
return (m_date);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const std::vector <const attachment*> messageParser::getAttachmentList() const
|
|
|
|
{
|
|
|
|
std::vector <const attachment*> res;
|
|
|
|
|
|
|
|
res.reserve(m_attach.size());
|
|
|
|
|
|
|
|
for (std::vector <attachment*>::const_iterator it = m_attach.begin() ;
|
|
|
|
it != m_attach.end() ; ++it)
|
|
|
|
{
|
|
|
|
res.push_back(*it);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (res);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const int messageParser::getAttachmentCount() const
|
|
|
|
{
|
|
|
|
return (m_attach.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const attachment* messageParser::getAttachmentAt(const int pos) const
|
|
|
|
{
|
|
|
|
return (m_attach[pos]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const std::vector <const textPart*> messageParser::getTextPartList() const
|
|
|
|
{
|
|
|
|
std::vector <const textPart*> res;
|
|
|
|
|
|
|
|
res.reserve(m_textParts.size());
|
|
|
|
|
|
|
|
for (std::vector <textPart*>::const_iterator it = m_textParts.begin() ;
|
|
|
|
it != m_textParts.end() ; ++it)
|
|
|
|
{
|
|
|
|
res.push_back(*it);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (res);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const int messageParser::getTextPartCount() const
|
|
|
|
{
|
|
|
|
return (m_textParts.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const textPart* messageParser::getTextPartAt(const int pos) const
|
|
|
|
{
|
|
|
|
return (m_textParts[pos]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-10-05 10:28:21 +00:00
|
|
|
} // vmime
|