vmime/src/messageParser.cpp
2013-01-10 17:30:31 +01:00

323 lines
7.2 KiB
C++

//
// VMime library (http://www.vmime.org)
// Copyright (C) 2002-2013 Vincent Richard <vincent@vmime.org>
//
// 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 3 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.,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
//
// Linking this library statically or dynamically with other modules is making
// a combined work based on this library. Thus, the terms and conditions of
// the GNU General Public License cover the whole combination.
//
#include "vmime/messageParser.hpp"
#include "vmime/attachmentHelper.hpp"
#include "vmime/defaultAttachment.hpp"
#include "vmime/textPartFactory.hpp"
#include "vmime/relay.hpp"
#include "vmime/contentTypeField.hpp"
#include "vmime/contentDispositionField.hpp"
namespace vmime
{
messageParser::messageParser(const string& buffer)
{
ref <message> msg = vmime::create <message>();
msg->parse(buffer);
parse(msg);
}
messageParser::messageParser(ref <const message> msg)
{
parse(msg);
}
messageParser::~messageParser()
{
}
void messageParser::parse(ref <const message> msg)
{
// Header fields (if field is present, copy its value, else do nothing)
#ifndef VMIME_BUILDING_DOC
#define TRY_FIELD(var, type, name) \
try { var = *msg->getHeader()->findField(name)->getValue().dynamicCast <type>(); } \
catch (exceptions::no_such_field) { }
TRY_FIELD(m_from, mailbox, fields::FROM);
TRY_FIELD(m_to, addressList, fields::TO);
TRY_FIELD(m_cc, addressList, fields::CC);
TRY_FIELD(m_bcc, addressList, fields::BCC);
TRY_FIELD(m_subject, text, fields::SUBJECT);
#undef TRY_FIELD
#endif // VMIME_BUILDING_DOC
// Date
try
{
const headerField& recv = *msg->getHeader()->findField(fields::RECEIVED);
m_date = recv.getValue().dynamicCast <const relay>()->getDate();
}
catch (vmime::exceptions::no_such_field&)
{
try
{
const headerField& date = *msg->getHeader()->findField(fields::DATE);
m_date = *date.getValue().dynamicCast <const datetime>();
}
catch (vmime::exceptions::no_such_field&)
{
m_date = datetime::now();
}
}
// Attachments
findAttachments(msg);
// Text parts
findTextParts(msg, msg);
}
void messageParser::findAttachments(ref <const message> msg)
{
m_attach = attachmentHelper::findAttachmentsInMessage(msg);
}
void messageParser::findTextParts(ref <const bodyPart> msg, ref <const bodyPart> part)
{
// Handle the case in which the message is not multipart: if the body part is
// "text/*", take this part.
if (part->getBody()->getPartCount() == 0)
{
mediaType type(mediaTypes::TEXT, mediaTypes::TEXT_PLAIN);
bool accept = false;
try
{
const contentTypeField& ctf = dynamic_cast<contentTypeField&>
(*msg->getHeader()->findField(fields::CONTENT_TYPE));
const mediaType ctfType =
*ctf.getValue().dynamicCast <const mediaType>();
if (ctfType.getType() == mediaTypes::TEXT)
{
type = ctfType;
accept = true;
}
}
catch (exceptions::no_such_field&)
{
// No "Content-type" field: assume "text/plain".
accept = true;
}
if (accept)
{
ref <textPart> txtPart = textPartFactory::getInstance()->create(type);
txtPart->parse(msg, msg, msg);
m_textParts.push_back(txtPart);
}
}
// Multipart message
else
{
findSubTextParts(msg, part);
}
}
bool messageParser::findSubTextParts(ref <const bodyPart> msg, ref <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 <ref <const bodyPart> > textParts;
for (size_t i = 0 ; i < part->getBody()->getPartCount() ; ++i)
{
const ref <const bodyPart> p = part->getBody()->getPartAt(i);
try
{
const contentTypeField& ctf = dynamic_cast <const contentTypeField&>
(*(p->getHeader()->findField(fields::CONTENT_TYPE)));
const mediaType type = *ctf.getValue().dynamicCast <const mediaType>();
contentDisposition disp; // default should be inline
if (type.getType() == mediaTypes::TEXT)
{
try
{
ref <const contentDispositionField> cdf = p->getHeader()->
findField(fields::CONTENT_DISPOSITION).dynamicCast <const contentDispositionField>();
disp = *cdf->getValue().dynamicCast <const contentDisposition>();
}
catch (exceptions::no_such_field&)
{
// No "Content-Disposition" field, assume default
}
if (disp.getName() == contentDispositionTypes::INLINE)
textParts.push_back(p);
}
}
catch (exceptions::no_such_field&)
{
// No "Content-type" field.
}
}
if (textParts.size())
{
// Okay. So we have found at least one text part
for (std::vector <ref <const bodyPart> >::const_iterator p = textParts.begin() ;
p != textParts.end() ; ++p)
{
const contentTypeField& ctf = dynamic_cast <const contentTypeField&>
(*((*p)->getHeader()->findField(fields::CONTENT_TYPE)));
const mediaType type = *ctf.getValue().dynamicCast <const mediaType>();
try
{
ref <textPart> txtPart = textPartFactory::getInstance()->create(type);
txtPart->parse(msg, part, *p);
m_textParts.push_back(txtPart);
}
catch (exceptions::no_factory_available& e)
{
// Content-type not recognized.
}
}
}
bool found = false;
for (size_t i = 0 ; !found && (i < part->getBody()->getPartCount()) ; ++i)
{
found = findSubTextParts(msg, part->getBody()->getPartAt(i));
}
return found;
}
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 <ref <const attachment> > messageParser::getAttachmentList() const
{
return m_attach;
}
size_t messageParser::getAttachmentCount() const
{
return (m_attach.size());
}
const ref <const attachment> messageParser::getAttachmentAt(const size_t pos) const
{
return (m_attach[pos]);
}
const std::vector <ref <const textPart> > messageParser::getTextPartList() const
{
std::vector <ref <const textPart> > res;
res.reserve(m_textParts.size());
for (std::vector <ref <textPart> >::const_iterator it = m_textParts.begin() ;
it != m_textParts.end() ; ++it)
{
res.push_back(*it);
}
return (res);
}
size_t messageParser::getTextPartCount() const
{
return (m_textParts.size());
}
const ref <const textPart> messageParser::getTextPartAt(const size_t pos) const
{
return (m_textParts[pos]);
}
} // vmime