diff options
Diffstat (limited to 'vmime-master/src/vmime/encoding.cpp')
-rw-r--r-- | vmime-master/src/vmime/encoding.cpp | 331 |
1 files changed, 331 insertions, 0 deletions
diff --git a/vmime-master/src/vmime/encoding.cpp b/vmime-master/src/vmime/encoding.cpp new file mode 100644 index 0000000..23814a6 --- /dev/null +++ b/vmime-master/src/vmime/encoding.cpp @@ -0,0 +1,331 @@ +// +// VMime library (http://www.vmime.org) +// Copyright (C) 2002 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/encoding.hpp" +#include "vmime/contentHandler.hpp" + +#include "vmime/utility/outputStreamStringAdapter.hpp" +#include "vmime/utility/encoder/encoderFactory.hpp" + +#include <algorithm> + + +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 <utility::encoder::encoder> encoding::getEncoder() const { + + shared_ptr <utility::encoder::encoder> 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<unsigned char>(), 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 <const contentHandler>& 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 <const contentHandler>& 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 <component> encoding::clone() const { + + return make_shared <encoding>(*this); +} + + +void encoding::copyFrom(const component& other) { + + const encoding& e = dynamic_cast <const encoding&>(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 <shared_ptr <component> > encoding::getChildComponents() { + + return std::vector <shared_ptr <component> >(); +} + + +} // vmime |