aboutsummaryrefslogtreecommitdiff
path: root/vmime-master/src/vmime/utility/stringUtils.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'vmime-master/src/vmime/utility/stringUtils.cpp')
-rw-r--r--vmime-master/src/vmime/utility/stringUtils.cpp337
1 files changed, 337 insertions, 0 deletions
diff --git a/vmime-master/src/vmime/utility/stringUtils.cpp b/vmime-master/src/vmime/utility/stringUtils.cpp
new file mode 100644
index 0000000..823a3ea
--- /dev/null
+++ b/vmime-master/src/vmime/utility/stringUtils.cpp
@@ -0,0 +1,337 @@
+//
+// 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/utility/stringUtils.hpp"
+#include "vmime/parserHelpers.hpp"
+
+
+namespace vmime {
+namespace utility {
+
+
+bool stringUtils::isStringEqualNoCase(const string& s1, const char* s2, const size_t n) {
+
+ // 'n' is the number of characters to compare
+ // 's2' must be in lowercase letters only
+ if (s1.length() < n) {
+ return (false);
+ }
+
+ const std::ctype <char>& fac =
+ std::use_facet <std::ctype <char> >(std::locale::classic());
+
+ bool equal = true;
+
+ for (size_t i = 0 ; equal && i < n ; ++i) {
+ equal = (fac.tolower(static_cast <unsigned char>(s1[i])) == s2[i]);
+ }
+
+ return equal;
+}
+
+
+bool stringUtils::isStringEqualNoCase(const string& s1, const string& s2) {
+
+ if (s1.length() != s2.length()) {
+ return false;
+ }
+
+ const std::ctype <char>& fac =
+ std::use_facet <std::ctype <char> >(std::locale::classic());
+
+ bool equal = true;
+ const string::const_iterator end = s1.end();
+
+ for (string::const_iterator i = s1.begin(), j = s2.begin() ; equal && i != end ; ++i, ++j) {
+ equal = (fac.tolower(static_cast <unsigned char>(*i)) == fac.tolower(static_cast <unsigned char>(*j)));
+ }
+
+ return equal;
+}
+
+
+bool stringUtils::isStringEqualNoCase(
+ const string::const_iterator begin,
+ const string::const_iterator end,
+ const char* s,
+ const size_t n
+) {
+
+ if (static_cast <size_t>(end - begin) < n) {
+ return false;
+ }
+
+ const std::ctype <char>& fac =
+ std::use_facet <std::ctype <char> >(std::locale::classic());
+
+ bool equal = true;
+ char* c = const_cast<char*>(s);
+ size_t r = n;
+
+ for (string::const_iterator i = begin ; equal && r && *c ; ++i, ++c, --r) {
+ equal = (fac.tolower(static_cast <unsigned char>(*i)) == static_cast <unsigned char>(*c));
+ }
+
+ return r == 0 && equal;
+}
+
+
+const string stringUtils::toLower(const string& str) {
+
+ const std::ctype <char>& fac =
+ std::use_facet <std::ctype <char> >(std::locale::classic());
+
+ string out;
+ out.resize(str.size());
+
+ for (size_t i = 0, len = str.length() ; i < len ; ++i) {
+ out[i] = fac.tolower(static_cast <unsigned char>(str[i]));
+ }
+
+ return out;
+}
+
+
+const string stringUtils::toUpper(const string& str) {
+
+ const std::ctype <char>& fac =
+ std::use_facet <std::ctype <char> >(std::locale::classic());
+
+ string out;
+ out.resize(str.size());
+
+ for (size_t i = 0, len = str.length() ; i < len ; ++i) {
+ out[i] = fac.toupper(static_cast <unsigned char>(str[i]));
+ }
+
+ return out;
+}
+
+
+const string stringUtils::trim(const string& str) {
+
+ string::const_iterator b = str.begin();
+ string::const_iterator e = str.end();
+
+ if (b != e) {
+
+ for ( ; b != e && parserHelpers::isSpace(*b) ; ++b) {}
+ for ( ; e != b && parserHelpers::isSpace(*(e - 1)) ; --e) {}
+ }
+
+ return string(b, e);
+}
+
+
+size_t stringUtils::countASCIIchars(
+ const string::const_iterator begin,
+ const string::const_iterator end
+) {
+
+ size_t count = 0;
+
+ for (string::const_iterator i = begin ; i != end ; ++i) {
+
+ if (parserHelpers::isAscii(*i)) {
+
+ if (*i != '=' || ((i + 1) != end && *(i + 1) != '?')) { // To avoid bad behaviour...
+ ++count;
+ }
+ }
+ }
+
+ return count;
+}
+
+
+bool stringUtils::is7bit(const string& str) {
+
+ return countASCIIchars(str.begin(), str.end()) == str.length();
+}
+
+
+size_t stringUtils::findFirstNonASCIIchar(
+ const string::const_iterator begin,
+ const string::const_iterator end
+) {
+
+ size_t pos = string::npos;
+
+ for (string::const_iterator i = begin ; i != end ; ++i) {
+
+ if (!parserHelpers::isAscii(*i)) {
+ pos = i - begin;
+ break;
+ }
+ }
+
+ return pos;
+}
+
+
+const string stringUtils::unquote(const string& str) {
+
+ if (str.length() < 2) {
+ return str;
+ }
+
+ if (str[0] != '"' || str[str.length() - 1] != '"') {
+ return str;
+ }
+
+ string res;
+ res.reserve(str.length());
+
+ bool escaped = false;
+
+ for (string::const_iterator it = str.begin() + 1, end = str.end() - 1 ; it != end ; ++it) {
+
+ const char c = *it;
+
+ if (escaped) {
+
+ res += c;
+ escaped = false;
+
+ } else if (!escaped && c == '\\') {
+
+ escaped = true;
+
+ } else {
+
+ res += c;
+ }
+ }
+
+ return res;
+}
+
+
+bool stringUtils::needQuoting(const string& str, const string& specialChars) {
+
+ return str.find_first_of(specialChars.c_str()) != string::npos;
+}
+
+
+string stringUtils::quote(
+ const string& str,
+ const string& escapeSpecialChars,
+ const string& escapeChar
+) {
+
+ std::ostringstream oss;
+ size_t lastPos = 0, pos = 0;
+
+ while ((pos = str.find_first_of(escapeSpecialChars, lastPos)) != string::npos) {
+
+ oss << str.substr(lastPos, pos - lastPos)
+ << escapeChar
+ << str[pos];
+
+ lastPos = pos + 1;
+ }
+
+ oss << str.substr(lastPos);
+
+ return oss.str();
+}
+
+
+bool stringUtils::isValidHostname(const vmime::string& hostname) {
+
+ short numberOfDots = 0;
+ return isValidFQDNImpl(hostname, &numberOfDots);
+}
+
+
+bool stringUtils::isValidFQDN(const vmime::string& fqdn) {
+
+ short numberOfDots = 0;
+ return isValidFQDNImpl(fqdn, &numberOfDots) && numberOfDots >= 2;
+}
+
+
+bool stringUtils::isValidFQDNImpl(const vmime::string& fqdn, short* numberOfDots) {
+
+ bool alphanumOnly = true;
+ bool invalid = false;
+ bool previousIsDot = true; // dot is not allowed as the first char
+ bool previousIsDash = true; // dash is not allowed as the first char
+
+ *numberOfDots = 0;
+
+ for (size_t i = 0, n = fqdn.length() ; alphanumOnly && !invalid && i < n ; ++i) {
+
+ const char c = fqdn[i];
+
+ alphanumOnly = (
+ (c >= '0' && c <= '9') // DIGIT
+ || (c >= 'a' && c <= 'z') // ALPHA
+ || (c >= 'A' && c <= 'Z') // ALPHA
+ || (c == '.')
+ || (c == '-')
+ );
+
+ if (c == '-') {
+
+ if (previousIsDot) {
+ invalid = true; // dash is not allowed as the first char
+ }
+
+ previousIsDot = false;
+ previousIsDash = true;
+
+ } else if (c == '.') {
+
+ if (previousIsDash) {
+
+ invalid = true; // dash is not allowed as the first char
+
+ } else if (previousIsDot) {
+
+ invalid = true; // consecutive dots are not allowed
+
+ } else {
+
+ ++*numberOfDots;
+ previousIsDot = true;
+ }
+
+ previousIsDash = false;
+
+ } else {
+
+ previousIsDot = false;
+ previousIsDash = false;
+ }
+ }
+
+ return alphanumOnly &&
+ !previousIsDot &&
+ !previousIsDash &&
+ !invalid;
+}
+
+
+} // utility
+} // vmime