aboutsummaryrefslogtreecommitdiff
path: root/vmime-master/src/vmime/encoding.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'vmime-master/src/vmime/encoding.cpp')
-rw-r--r--vmime-master/src/vmime/encoding.cpp331
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