// // VMime library (http://www.vmime.org) // Copyright (C) 2002 Vincent Richard // // 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/config.hpp" #if VMIME_HAVE_MESSAGING_FEATURES #include "vmime/net/transport.hpp" #include "vmime/utility/stream.hpp" #include "vmime/mailboxList.hpp" #include "vmime/message.hpp" #include "vmime/dateTime.hpp" #include "vmime/messageId.hpp" #include "vmime/utility/outputStreamAdapter.hpp" #include "vmime/utility/inputStreamStringAdapter.hpp" namespace vmime { namespace net { transport::transport( const shared_ptr & sess, const serviceInfos& infos, const shared_ptr & auth ) : service(sess, infos, auth) { } shared_ptr transport::processHeaderField(const shared_ptr & field) { if (utility::stringUtils::isStringEqualNoCase(field->getName(), fields::BCC)) { // Remove Bcc headers from the message, as required by the RFC. // Some SMTP server automatically strip this header (Postfix, qmail), // and others have an option for this (Exim). return null; } else if (utility::stringUtils::isStringEqualNoCase(field->getName(), fields::RETURN_PATH)) { // RFC-2821: Return-Path header is added by the final transport system // that delivers the message to its recipient. Then, it should not be // transmitted to MSA. return null; } else if (utility::stringUtils::isStringEqualNoCase(field->getName(), fields::ORIGINAL_RECIPIENT)) { // RFC-2298: Delivering MTA may add the Original-Recipient header and // discard existing one; so, no need to send it. return null; } // Leave the header field as is return field; } void transport::processHeader(const shared_ptr
& header) { if (header->getFieldCount() == 0) { return; } // Remove/replace fields for (size_t idx = header->getFieldCount() ; idx != 0 ; --idx) { shared_ptr field = header->getFieldAt(idx - 1); shared_ptr newField = processHeaderField(field); if (newField == NULL) { header->removeField(field); } else if (newField != field) { header->replaceField(field, newField); } } // Add missing header fields // -- Date if (!header->hasField(fields::DATE)) { header->Date()->setValue(datetime::now()); } // -- Mime-Version if (!header->hasField(fields::MIME_VERSION)) { header->MimeVersion()->setValue(string(SUPPORTED_MIME_VERSION)); } // -- Message-Id if (!header->hasField(fields::MESSAGE_ID)) { header->MessageId()->setValue(messageId::generateId()); } } static void extractMailboxes( mailboxList& recipients, const addressList& list ) { for (size_t i = 0 ; i < list.getAddressCount() ; ++i) { shared_ptr mbox = dynamicCast (list.getAddressAt(i)->clone()); if (mbox) { recipients.appendMailbox(mbox); } } } void transport::send( const shared_ptr & msg, utility::progressListener* progress, const dsnAttributes& dsnAttrs ) { // Extract expeditor shared_ptr fromMbox = msg->getHeader()->findFieldValue (fields::FROM); if (!fromMbox) { throw exceptions::no_expeditor(); } mailbox expeditor = *fromMbox; // Extract sender shared_ptr senderMbox = msg->getHeader()->findFieldValue (fields::SENDER); mailbox sender; if (!senderMbox) { sender = expeditor; } else { sender = *senderMbox; } // Extract recipients mailboxList recipients; // -- "To" field shared_ptr addresses = msg->getHeader()->findFieldValue (fields::TO); if (addresses) { extractMailboxes(recipients, *addresses); } // -- "Cc" field addresses = msg->getHeader()->findFieldValue (fields::CC); if (addresses) { extractMailboxes(recipients, *addresses); } // -- "Bcc" field addresses = msg->getHeader()->findFieldValue (fields::BCC); if (addresses) { extractMailboxes(recipients, *addresses); } // Process message header by removing fields that should be removed // before transmitting the message to MSA, and adding missing fields // which are required/recommended by the RFCs. shared_ptr
hdr = vmime::clone(msg->getHeader()); processHeader(hdr); // To avoid cloning message body (too much overhead), use processed // header during the time we are generating the message to a stream. // Revert it back to original header after. struct XChangeMsgHeader { XChangeMsgHeader( const shared_ptr & _msg, const shared_ptr & _hdr ) : msg(_msg), hdr(msg->getHeader()) { // Set new header msg->setHeader(_hdr); } ~XChangeMsgHeader() { // Revert original header msg->setHeader(hdr); } private: shared_ptr msg; shared_ptr hdr; } headerExchanger(msg, hdr); send(msg, expeditor, recipients, progress, sender, dsnAttrs); } void transport::send( const shared_ptr & msg, const mailbox& expeditor, const mailboxList& recipients, utility::progressListener* progress, const mailbox& sender, const dsnAttributes& dsnAttrs ) { // Generate the message, "stream" it and delegate the sending // to the generic send() function. std::ostringstream oss; utility::outputStreamAdapter ossAdapter(oss); msg->generate(ossAdapter); const string& str(oss.str()); utility::inputStreamStringAdapter isAdapter(str); send(expeditor, recipients, isAdapter, str.length(), progress, sender, dsnAttrs); } transport::Type transport::getType() const { return TYPE_TRANSPORT; } } // net } // vmime #endif // VMIME_HAVE_MESSAGING_FEATURES