// // 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/encoding.hpp" #include "vmime/contentHandler.hpp" #include "vmime/utility/outputStreamStringAdapter.hpp" #include "vmime/utility/encoder/encoderFactory.hpp" #include namespace vmime { encoding::encoding() : m_name(encodingTypes::SEVEN_BIT), m_usage(USAGE_UNKNOWN) { } encoding::encoding(const string& name) : m_name(utility::stringUtils::toLower(name)), m_usage(USAGE_UNKNOWN) { } encoding::encoding(const string& name, const EncodingUsage usage) : m_name(utility::stringUtils::toLower(name)), m_usage(usage) { } encoding::encoding(const encoding& enc) : headerFieldValue(), m_name(enc.m_name), m_usage(enc.m_usage) { } void encoding::parseImpl( const parsingContext& /* ctx */, const string& buffer, const size_t position, const size_t end, size_t* newPosition ) { m_usage = USAGE_UNKNOWN; m_name = utility::stringUtils::toLower( utility::stringUtils::trim( utility::stringUtils::unquote( utility::stringUtils::trim( string(buffer.begin() + position, buffer.begin() + end) ) ) ) ); if (m_name.empty()) { m_name = encodingTypes::SEVEN_BIT; // assume default "7-bit" } setParsedBounds(position, end); if (newPosition) { *newPosition = end; } } void encoding::generateImpl( const generationContext& /* ctx */, utility::outputStream& os, const size_t curLinePos, size_t* newLinePos ) const { os << m_name; if (newLinePos) { *newLinePos = curLinePos + m_name.length(); } } shared_ptr encoding::getEncoder() const { shared_ptr encoder = utility::encoder::encoderFactory::getInstance()->create(generate()); // FIXME: this should not be here (move me into QP encoder instead?) if (m_usage == USAGE_TEXT && m_name == encodingTypes::QUOTED_PRINTABLE) { encoder->getProperties()["text"] = true; } return encoder; } encoding& encoding::operator=(const encoding& other) { copyFrom(other); return (*this); } encoding& encoding::operator=(const string& name) { m_name = utility::stringUtils::toLower(name); m_usage = USAGE_UNKNOWN; return *this; } bool encoding::operator==(const encoding& value) const { return utility::stringUtils::toLower(m_name) == value.m_name; } bool encoding::operator!=(const encoding& value) const { return !(*this == value); } const encoding encoding::decideImpl( const string::const_iterator begin, const string::const_iterator end ) { const string::difference_type length = end - begin; const string::difference_type count = std::count_if( begin, end, std::bind2nd(std::less(), 127) ); // All is in 7-bit US-ASCII --> 7-bit (or Quoted-Printable...) if (length == count) { // Now, we check if there is any line with more than // "lineLengthLimits::convenient" characters (7-bit requires that) string::const_iterator p = begin; const size_t maxLen = lineLengthLimits::convenient; size_t len = 0; for ( ; p != end && len <= maxLen ; ) { if (*p == '\n') { len = 0; ++p; // May or may not need to be encoded, we don't take // any risk (avoid problems with SMTP) if (p != end && *p == '.') { len = maxLen + 1; } } else { ++len; ++p; } } if (len > maxLen) { return encoding(encodingTypes::QUOTED_PRINTABLE); } else { return encoding(encodingTypes::SEVEN_BIT); } // Less than 20% non US-ASCII --> Quoted-Printable } else if ((length - count) <= length / 5) { return encoding(encodingTypes::QUOTED_PRINTABLE); // Otherwise --> Base64 } else { return encoding(encodingTypes::BASE64); } } bool encoding::shouldReencode() const { if (m_name == encodingTypes::BASE64 || m_name == encodingTypes::QUOTED_PRINTABLE || m_name == encodingTypes::UUENCODE) { return false; } return true; } const encoding encoding::decide( const shared_ptr & data, const EncodingUsage usage ) { // Do not re-encode data if it is already encoded if (data->isEncoded() && !data->getEncoding().shouldReencode()) { return data->getEncoding(); } encoding enc; if (usage == USAGE_TEXT && data->isBuffered() && data->getLength() > 0 && data->getLength() < 32768) { // Extract data into temporary buffer string buffer; utility::outputStreamStringAdapter os(buffer); data->extract(os); os.flush(); enc = decideImpl(buffer.begin(), buffer.end()); } else { enc = encoding(encodingTypes::BASE64); } enc.setUsage(usage); return enc; } const encoding encoding::decide( const shared_ptr & data, const charset& chset, const EncodingUsage usage ) { // Do not re-encode data if it is already encoded if (data->isEncoded() && !data->getEncoding().shouldReencode()) { return data->getEncoding(); } if (usage == USAGE_TEXT) { encoding recEncoding; if (chset.getRecommendedEncoding(recEncoding)) { recEncoding.setUsage(usage); return recEncoding; } } return decide(data, usage); } shared_ptr encoding::clone() const { return make_shared (*this); } void encoding::copyFrom(const component& other) { const encoding& e = dynamic_cast (other); m_name = e.m_name; } const string& encoding::getName() const { return m_name; } void encoding::setName(const string& name) { m_name = name; } encoding::EncodingUsage encoding::getUsage() const { return m_usage; } void encoding::setUsage(const EncodingUsage usage) { m_usage = usage; } const std::vector > encoding::getChildComponents() { return std::vector >(); } } // vmime