2005-10-13 21:27:46 +00:00
|
|
|
//
|
|
|
|
// VMime library (http://www.vmime.org)
|
2006-02-05 10:22:59 +00:00
|
|
|
// Copyright (C) 2002-2006 Vincent Richard <vincent@vincent-richard.net>
|
2005-10-13 21:27:46 +00:00
|
|
|
//
|
|
|
|
// 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.,
|
|
|
|
// 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/attachmentHelper.hpp"
|
|
|
|
|
|
|
|
#include "vmime/bodyPartAttachment.hpp"
|
2006-01-16 21:55:37 +00:00
|
|
|
#include "vmime/parsedMessageAttachment.hpp"
|
|
|
|
#include "vmime/generatedMessageAttachment.hpp"
|
|
|
|
|
2005-11-04 23:21:22 +00:00
|
|
|
#include "vmime/disposition.hpp"
|
2006-01-16 21:55:37 +00:00
|
|
|
#include "vmime/emptyContentHandler.hpp"
|
2005-10-13 21:27:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace vmime
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
// static
|
|
|
|
const bool attachmentHelper::isBodyPartAnAttachment(ref <const bodyPart> part)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const contentDispositionField& cdf = dynamic_cast<contentDispositionField&>
|
|
|
|
(*part->getHeader()->findField(fields::CONTENT_DISPOSITION));
|
|
|
|
|
2005-11-04 23:21:22 +00:00
|
|
|
const contentDisposition disp = *cdf.getValue()
|
|
|
|
.dynamicCast <const contentDisposition>();
|
|
|
|
|
|
|
|
if (disp.getName() != contentDispositionTypes::INLINE)
|
2005-10-13 21:27:46 +00:00
|
|
|
return true;
|
2006-01-19 19:58:37 +00:00
|
|
|
|
|
|
|
// If the Content-Disposition is 'inline' and there is no
|
|
|
|
// Content-Id or Content-Location field, it is an attachment
|
|
|
|
if (!part->getHeader()->hasField(vmime::fields::CONTENT_ID) &&
|
|
|
|
!part->getHeader()->hasField(vmime::fields::CONTENT_LOCATION))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2005-10-13 21:27:46 +00:00
|
|
|
}
|
2005-11-04 23:21:22 +00:00
|
|
|
catch (exceptions::no_such_field&)
|
2005-10-13 21:27:46 +00:00
|
|
|
{
|
|
|
|
// No "Content-disposition" field: assume "attachment" if
|
|
|
|
// type is not "text/..." or "multipart/...".
|
|
|
|
mediaType type;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const contentTypeField& ctf = dynamic_cast<contentTypeField&>
|
|
|
|
(*part->getHeader()->findField(fields::CONTENT_TYPE));
|
|
|
|
|
2005-11-04 23:21:22 +00:00
|
|
|
type = *ctf.getValue().dynamicCast <const mediaType>();
|
2005-10-13 21:27:46 +00:00
|
|
|
}
|
2005-11-04 23:21:22 +00:00
|
|
|
catch (exceptions::no_such_field&)
|
2005-10-13 21:27:46 +00:00
|
|
|
{
|
|
|
|
// No "Content-type" field: assume "application/octet-stream".
|
|
|
|
type = mediaType(mediaTypes::APPLICATION,
|
|
|
|
mediaTypes::APPLICATION_OCTET_STREAM);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type.getType() != mediaTypes::TEXT &&
|
|
|
|
type.getType() != mediaTypes::MULTIPART)
|
|
|
|
{
|
2005-10-14 18:21:43 +00:00
|
|
|
// If a "Content-Id" field is present, it might be an
|
|
|
|
// embedded object (MHTML messages)
|
|
|
|
if (part->getHeader()->hasField(vmime::fields::CONTENT_ID))
|
|
|
|
return false;
|
|
|
|
|
2005-10-13 21:27:46 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// static
|
|
|
|
ref <const attachment>
|
|
|
|
attachmentHelper::getBodyPartAttachment(ref <const bodyPart> part)
|
|
|
|
{
|
|
|
|
if (!isBodyPartAnAttachment(part))
|
|
|
|
return NULL;
|
|
|
|
|
2006-01-16 21:55:37 +00:00
|
|
|
mediaType type;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const contentTypeField& ctf = dynamic_cast<contentTypeField&>
|
|
|
|
(*part->getHeader()->findField(fields::CONTENT_TYPE));
|
|
|
|
|
|
|
|
type = *ctf.getValue().dynamicCast <const mediaType>();
|
|
|
|
}
|
|
|
|
catch (exceptions::no_such_field&)
|
|
|
|
{
|
|
|
|
// No "Content-type" field: assume "application/octet-stream".
|
|
|
|
type = mediaType(mediaTypes::APPLICATION,
|
|
|
|
mediaTypes::APPLICATION_OCTET_STREAM);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type.getType() == mediaTypes::MESSAGE &&
|
|
|
|
type.getSubType() == mediaTypes::MESSAGE_RFC822)
|
|
|
|
{
|
|
|
|
return vmime::create <generatedMessageAttachment>(part);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return vmime::create <bodyPartAttachment>(part);
|
|
|
|
}
|
2005-10-13 21:27:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// static
|
|
|
|
const std::vector <ref <const attachment> >
|
|
|
|
attachmentHelper::findAttachmentsInMessage(ref <const message> msg)
|
|
|
|
{
|
|
|
|
return findAttachmentsInBodyPart(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// static
|
|
|
|
const std::vector <ref <const attachment> >
|
|
|
|
attachmentHelper::findAttachmentsInBodyPart(ref <const bodyPart> part)
|
|
|
|
{
|
|
|
|
std::vector <ref <const attachment> > atts;
|
|
|
|
|
|
|
|
// Test this part
|
|
|
|
if (isBodyPartAnAttachment(part))
|
|
|
|
{
|
|
|
|
atts.push_back(getBodyPartAttachment(part));
|
|
|
|
}
|
|
|
|
// Find in sub-parts
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ref <const body> bdy = part->getBody();
|
|
|
|
|
|
|
|
for (int i = 0 ; i < bdy->getPartCount() ; ++i)
|
|
|
|
{
|
|
|
|
std::vector <ref <const attachment> > partAtts =
|
|
|
|
findAttachmentsInBodyPart(bdy->getPartAt(i));
|
|
|
|
|
2006-11-14 13:19:57 +00:00
|
|
|
std::copy(partAtts.begin(), partAtts.end(), std::back_inserter(atts));
|
2005-10-13 21:27:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return atts;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// static
|
|
|
|
void attachmentHelper::addAttachment(ref <message> msg, ref <attachment> att)
|
|
|
|
{
|
|
|
|
// We simply search for a "multipart/mixed" part. If no one exists,
|
|
|
|
// create it in the root part. This (very simple) algorithm should
|
|
|
|
// work in the most cases.
|
|
|
|
|
|
|
|
vmime::mediaType mpMixed(vmime::mediaTypes::MULTIPART,
|
|
|
|
vmime::mediaTypes::MULTIPART_MIXED);
|
|
|
|
|
|
|
|
ref <bodyPart> part = findBodyPart(msg, mpMixed);
|
|
|
|
|
|
|
|
if (part == NULL) // create it
|
|
|
|
{
|
2006-01-16 21:55:37 +00:00
|
|
|
if (msg->getBody()->getPartCount() != 0)
|
2005-10-13 21:27:46 +00:00
|
|
|
{
|
2006-01-16 21:55:37 +00:00
|
|
|
// Create a new container part for the parts that were in
|
|
|
|
// the root part of the message
|
|
|
|
ref <bodyPart> container = vmime::create <bodyPart>();
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if (msg->getHeader()->hasField(fields::CONTENT_TYPE))
|
|
|
|
{
|
|
|
|
container->getHeader()->ContentType()->setValue
|
|
|
|
(msg->getHeader()->ContentType()->getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msg->getHeader()->hasField(fields::CONTENT_TRANSFER_ENCODING))
|
|
|
|
{
|
|
|
|
container->getHeader()->ContentTransferEncoding()->setValue
|
|
|
|
(msg->getHeader()->ContentTransferEncoding()->getValue());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (exceptions::no_such_field&)
|
|
|
|
{
|
|
|
|
// Ignore
|
|
|
|
}
|
|
|
|
|
|
|
|
// Move parts from the root part to this new part
|
|
|
|
const std::vector <ref <bodyPart> > partList =
|
|
|
|
msg->getBody()->getPartList();
|
|
|
|
|
|
|
|
msg->getBody()->removeAllParts();
|
|
|
|
|
|
|
|
for (unsigned int i = 0 ; i < partList.size() ; ++i)
|
|
|
|
container->getBody()->appendPart(partList[i]);
|
|
|
|
|
|
|
|
msg->getBody()->appendPart(container);
|
2005-10-13 21:27:46 +00:00
|
|
|
}
|
2006-01-16 21:55:37 +00:00
|
|
|
else
|
2005-10-13 21:27:46 +00:00
|
|
|
{
|
2006-01-16 21:55:37 +00:00
|
|
|
// The message is a simple (RFC-822) message, and do not
|
|
|
|
// contains any MIME part. Move the contents from the
|
|
|
|
// root to a new child part.
|
|
|
|
ref <bodyPart> child = vmime::create <bodyPart>();
|
|
|
|
|
|
|
|
if (msg->getHeader()->hasField(fields::CONTENT_TYPE))
|
|
|
|
{
|
|
|
|
child->getHeader()->ContentType()->setValue
|
|
|
|
(msg->getHeader()->ContentType()->getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msg->getHeader()->hasField(fields::CONTENT_TRANSFER_ENCODING))
|
|
|
|
{
|
|
|
|
child->getHeader()->ContentTransferEncoding()->setValue
|
|
|
|
(msg->getHeader()->ContentTransferEncoding()->getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
child->getBody()->setContents(msg->getBody()->getContents());
|
|
|
|
msg->getBody()->setContents(vmime::create <emptyContentHandler>());
|
|
|
|
|
|
|
|
msg->getBody()->appendPart(child);
|
2005-10-13 21:27:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set the root part to 'multipart/mixed'
|
|
|
|
msg->getHeader()->ContentType()->setValue(mpMixed);
|
|
|
|
|
2006-01-16 21:55:37 +00:00
|
|
|
msg->getHeader()->removeAllFields(vmime::fields::CONTENT_DISPOSITION);
|
|
|
|
msg->getHeader()->removeAllFields(vmime::fields::CONTENT_TRANSFER_ENCODING);
|
2005-10-13 21:27:46 +00:00
|
|
|
|
|
|
|
part = msg;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate the attachment part
|
2006-07-14 08:28:35 +00:00
|
|
|
att->generateIn(part);
|
2005-10-13 21:27:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// static
|
|
|
|
ref <bodyPart> attachmentHelper::findBodyPart
|
|
|
|
(ref <bodyPart> part, const mediaType& type)
|
|
|
|
{
|
|
|
|
if (part->getBody()->getContentType() == type)
|
|
|
|
return part;
|
|
|
|
|
|
|
|
// Try in sub-parts
|
|
|
|
ref <body> bdy = part->getBody();
|
|
|
|
|
|
|
|
for (int i = 0 ; i < bdy->getPartCount() ; ++i)
|
|
|
|
{
|
|
|
|
ref <bodyPart> found =
|
|
|
|
findBodyPart(bdy->getPartAt(i), type);
|
|
|
|
|
|
|
|
if (found != NULL)
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-01-16 21:55:37 +00:00
|
|
|
// static
|
|
|
|
void attachmentHelper::addAttachment(ref <message> msg, ref <message> amsg)
|
|
|
|
{
|
|
|
|
ref <attachment> att = vmime::create <parsedMessageAttachment>(amsg);
|
|
|
|
addAttachment(msg, att);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-10-13 21:27:46 +00:00
|
|
|
} // vmime
|
|
|
|
|