diff --git a/src/bash/filter.sh b/src/bash/filter.sh index de62fe7..e62118e 100755 --- a/src/bash/filter.sh +++ b/src/bash/filter.sh @@ -34,7 +34,8 @@ ########################################################################## ########################################################################## -# - installer l' utilitaire apg pour génération de mot de passes +# - installer l'utilitaire apg pour génération de mot de passes +# - installer l'utilitaire dos2unix # - le contenu de INSPECT_DIR doit être accessible en écriture pour le # proriétaire du script # - shrinkEMail et jirafeau.sh doivent être accessible en execution pour @@ -54,6 +55,7 @@ EX_TOO_LARGE=552 INSPECT_DIR=/var/spool/filter DIR_LOG=/var/log/mail FIC_LOG="${DIR_LOG}/filter.log" +TMP_LOG="$(mktemp)" SENDMAIL="/usr/sbin/sendmail -G -i" MAILS=/tmp/FILTER MAX_KEEP_IN_MAIL=5ki @@ -85,11 +87,13 @@ NL=' #--------------------- Fichier de LOG ------------------- LOG_FIC () { - echo "${BLUE}$(date +%d-%m-%Y-%H-%M-%S)${NC} : $*" >> "${FIC_LOG}" + echo "${BLUE}$(date +%d-%m-%Y-%H-%M-%S)${NC} : $*" >> "${TMP_LOG}" } quitFilter () { LOG_FIC "${GREEN}######################################## filter stop${NC}" + cat "${TMP_LOG}" >> "${FIC_LOG}" + rm -f "${TMP_LOG}" exit $1 } @@ -100,8 +104,8 @@ keepFailed () { } #################### MAIN ################################################# -echo "${NL}" >> "${FIC_LOG}" -LOG_FIC "${GREEN}######################################## filter start${NC}" +echo "${NL}${BLUE}$(date +%d-%m-%Y-%H-%M-%S)${NC} : ${GREEN}######################################## filter start (log in ${TMP_LOG})${NC}" >> "${FIC_LOG}" +LOG_FIC "${GREEN}######################################## ${TMP_LOG} ${NC}" if ! mkdir -p "${MAILS}"; then LOG_FIC "${RED}Can't mkdir ${MAILS} ${NC}" diff --git a/src/bash/filterTest.sh b/src/bash/filterTest.sh old mode 100644 new mode 100755 index e9ca941..1a4c05c --- a/src/bash/filterTest.sh +++ b/src/bash/filterTest.sh @@ -1,4 +1,37 @@ #!/bin/bash +########################################################################## +# Copyright KAZ 2021 # +# # +# contact (at) kaz.bzh # +# # +# This software is a filter to shrink email by attachment extraction. # +# # +# This software is governed by the CeCILL-B license under French law and # +# abiding by the rules of distribution of free software. You can use, # +# modify and/or redistribute the software under the terms of the # +# CeCILL-B license as circulated by CEA, CNRS and INRIA at the following # +# URL "http://www.cecill.info". # +# # +# As a counterpart to the access to the source code and rights to copy, # +# modify and redistribute granted by the license, users are provided # +# only with a limited warranty and the software's author, the holder of # +# the economic rights, and the successive licensors have only limited # +# liability. # +# # +# In this respect, the user's attention is drawn to the risks associated # +# with loading, using, modifying and/or developing or reproducing the # +# software by the user in light of its specific status of free software, # +# that may mean that it is complicated to manipulate, and that also # +# therefore means that it is reserved for developers and experienced # +# professionals having in-depth computer knowledge. Users are therefore # +# encouraged to load and test the software's suitability as regards # +# their requirements in conditions enabling the security of their # +# systems and/or data to be ensured and, more generally, to use and # +# operate it in the same conditions as regards security. # +# # +# The fact that you are presently reading this means that you have had # +# knowledge of the CeCILL-B license and that you accept its terms. # +########################################################################## PRG=$(basename $0) @@ -16,21 +49,16 @@ NL=' ' TTY=$(tty) - - ######################################## LOG () { echo "$1" >> "${TTY}" } usage () { - echo "Usage: ${PRG} mbox" + echo "Usage: ${PRG} [-h|-v|-g] [-m {NONE|FOOTER|ATTACHMENT|BOTH}] mbox" exit 1 } -[ "$#" -eq 1 ] || usage - - ######################################## mbox=$(realpath $1) dos2unix "${mbox}" @@ -41,23 +69,48 @@ JIRAFEAU_LOCAL="${JIRAFEAU_URL}" TMP_DIR="$(mktemp)" +######################################## +# recherche des binaires eMailShrinker="$(realpath "./eMailShrinker")" [ -x "${eMailShrinker}" ] || eMailShrinker="$(realpath "../../build/out/eMailShrinker")" +[ -x "${eMailShrinker}" ] || ( echo "${RED}eMailShrinker not found${NC}" ; exit) jirafeauAPI="$(realpath "./jirafeauAPI")" [ -x "${jirafeauAPI}" ] || jirafeauAPI="$(realpath "../../build/out/jirafeauAPI")" +[ -x "${jirafeauAPI}" ] || ( echo "${RED}jirafeauAPI not found${NC}" ; exit) + +for ARG in $*; do + case "$1" in + -h*) usage;; + -v*) "${eMailShrinker}" -v; exit;; + -g) DEBUG="-g"; shift;; + -m) shift; ATTACH_MODE="-m $1"; shift;; # XXX test option + *) break;; + esac +done +case "${ATTACH_MODE}" in + NONE|FOOTER|ATTACHMENT|BOTH);; + *) usage;; +esac + +[ "$#" -eq 1 ] || usage +######################################## +# nettoyage rm -f "${TMP_DIR}" ; mkdir -p "${TMP_DIR}" rm -fr "${TMP_DIR}/PJ-name.txt" "${TMP_DIR}/PJ-Keys.txt" "${TMP_DIR}/PJ" "${TMP_DIR}/archive-content.txt" "${TMP_DIR}/url-to-refresh.txt" "${TMP_DIR}/new-mbox" + echo -e "time: $(date "+%Y-%m-%d-%H:%M:%S")\nid: $(date +%s)" > "${TMP_DIR}/archive-content.txt" +######################################## +# affichage de la structure de départ LOG " - ${BLUE}mbox: ${mbox}${NC}" "${eMailShrinker}" -l "${mbox}" LOG +######################################## +# recherche des prolongations des délais de grace "${eMailShrinker}" -u "${mbox}" > "${TMP_DIR}/url-to-refresh.txt" 2>> "${TTY}" - - cat "${TMP_DIR}/url-to-refresh.txt" | grep "${JIRAFEAU_URL}" | while read REMOTE_LINK; do REMOTE_REF=$(echo "${REMOTE_LINK}" | sed -e 's/.*h=\([^&]*\).*/\1/' -e 's/.*http.*//') [ -z "${REMOTE_REF}" ] && continue @@ -67,12 +120,16 @@ cat "${TMP_DIR}/url-to-refresh.txt" | grep "${JIRAFEAU_URL}" | while read REMOTE echo "old: ${REMOTE_REF} ${REMOTE_KEY}" >> "${TMP_DIR}/archive-content.txt" done +######################################## +# extraction des pièces jointes "${eMailShrinker}" -s "5ki" -d "${TMP_DIR}/PJ" "${mbox}" > "${TMP_DIR}/PJ-name.txt" LOG " - ${BLUE}PJ-name: ${NC}" cat "${TMP_DIR}/PJ-name.txt" LOG +######################################## +# dépot des extractions dans jirafeau et récupération des codes cat "${TMP_DIR}/PJ-name.txt" | { while read ATTACH_TMP_NAME; do LOG " - ${BLUE}find ATTACH_TMP_NAME: (${ATTACH_TMP_NAME}) ${NC}" @@ -138,13 +195,13 @@ LOG LOG " - ${GREEN}ATTACH_MODE: ${ATTACH_MODE}${NC}" +######################################## +# substitution des pièces jointes par les codes fournis par jirafeau cat "${TMP_DIR}/PJ-Keys.txt" | "${eMailShrinker}" ${ATTACH_MODE} -s "5ki" "${mbox}" "${TMP_DIR}/new-mbox" 2>> "${TTY}" - +######################################## +# affichage de la structure à la fin LOG " - ${BLUE}new-mbox:${NC}" "${eMailShrinker}" -l "${TMP_DIR}/new-mbox" 2>> "${TTY}" -#echo "${TMP_DIR}" -#find "${TMP_DIR}" -type f -print - echo -e "\nresul in ${TMP_DIR}/new-mbox" diff --git a/src/cpp/Attachment.cpp b/src/cpp/Attachment.cpp index 9a365b9..7a2ba77 100644 --- a/src/cpp/Attachment.cpp +++ b/src/cpp/Attachment.cpp @@ -65,17 +65,9 @@ const string Attachment::ALTERNATIVE ("alternative"); const string Attachment::KAZ_ATTACH_NAME (".---KazAttachment---.html"); const string Attachment::MULTIPART ("multipart/"); - -const regex Attachment::nameCharsetRegEx (".*name\\*=[ \t]*(.*)"); -const regex Attachment::nameRegEx (".*name=[ \t]*((\"(\\\\.|[^\\\\\r])*\")|[^\r; ]*);?.*"); -// boundary="----=_Part_796779_1154936629.1668080348646" -// boundary="------------040709000505010508040808" -// boundary="----------=_1668606031-941125-91" -// boundary="_004_PAVPR10MB6792713B313048E3A259B215B2079PAVPR10MB6792EURP_"; -// boundary="_000_PAVPR10MB6792713B313048E3A259B215B2079PAVPR10MB6792EURP_" -// boundary=--boundary_1351_64006126-2b0e-4a3b-98ac-4797d1634188 -// boundary=--boundary_1352_7e294c9a-cfab-44a0-bfb3-7310380ac7cb; -const regex Attachment::boundaryRegEx (".*boundary=[ \t]*((\"(\\\\.|[^\\\\])*\")|[^; ]*);?.*"); +const regex Attachment::nameCharsetRegEx (".*name\\*=\\s*([; \t]*)"); +const regex Attachment::nameRegEx ( ".*name=\\s*((\"(\\\\.|[^\\\\])*\")|[^; \t]*).*"); +const regex Attachment::boundaryRegEx (".*boundary=\\s*((\"(\\\\.|[^\\\\])*\")|[^; \t]*).*"); const regex Attachment::cidDefRegEx (".*<([^>]*)>.*"); const regex Attachment::textRegEx (".*text/("+PLAIN+"|"+HTML+").*"); const regex Attachment::multiRegEx ("\\s*"+MULTIPART+"(mixed|"+RELATED+"|"+ALTERNATIVE+"|"+SIGNED+").*"); @@ -188,23 +180,49 @@ Attachment::getAttachName () const { static string tokens [] = {contentTypeToken, contentDispositionToken}; DEF_LOG ("Attachment::getAttachName", ""); for (string token : tokens) { + // name= string result = getProp (token, nameRegEx); removeQuote (result); if (result.length ()) { LOG ("name=: " << result); - encodedWord (result); + encodedWordDecode (result); + return result; + } + // name*x= + for (int id = 0; ; ++id) { + string item = getProp (token, regex (".*name\\*"+to_string (id)+"=\\s*((\"(\\\\.|[^\\\\])*\")|[; \t]*).*")); + if (item.empty ()) + break; + result += item; + } + removeQuote (result); + if (result.length ()) { + LOG ("name*x=: " << result); + encodedWordDecode (result); return result; } + // name*= result = getProp (token, nameCharsetRegEx); removeQuote (result); if (result.length ()) { LOG ("name*=: " << result); - charsetValue (result); + charsetValueDecode (result); + return result; + } + // name*x*= + for (int id = 0; ; ++id) { + string item = getProp (token, regex (".*name\\*"+to_string (id)+"\\*=\\s*([^; ]*)")); + if (item.empty ()) + break; + result += item; + } + removeQuote (result); + if (result.length ()) { + LOG ("name*x*=: " << result); + encodedWordDecode (result); return result; } } - // XXX il faut composer s'il y a plusieurs ligne filename*x= - // XXX il faut composer s'il y a plusieurs ligne filename*x*= return getUnknown (getContentType ()); } @@ -257,7 +275,7 @@ Attachment::isDefProp (const string &token, const string &val) const { if (it == env.end ()) return false; // XXX case insensitive ?? - return it->second.find (val) != string::npos; + return caseInsensitiveFind (it->second, val) != string::npos; } // ================================================================================ @@ -312,7 +330,8 @@ Attachment::readMime (ifstream &mbox, streamoff &curPos) { lastVar = line.substr (0, colonPos); toLower (lastVar); LOG ("find var: " << lastVar); - string val (cleanString (line.length () >= colonPos+2 ? line.substr (colonPos+2) : "")); // XXX check RFC " " after ": " + // XXX check in RFC if " " after ": " (=> +2 or +1) + string val (cleanString (line.length () >= colonPos+2 ? line.substr (colonPos+2) : "")); LOG ("new var: <" << lastVar << " <=> " << val << ">"); env [lastVar] = val; } diff --git a/src/cpp/MainAttachment.cpp b/src/cpp/MainAttachment.cpp index c534d45..665c354 100644 --- a/src/cpp/MainAttachment.cpp +++ b/src/cpp/MainAttachment.cpp @@ -61,7 +61,7 @@ static const string TMPL_FILENAME ("{{FILENAME}}"); static const string CID ("cid:"); // "l=/" => v1 compatibility -static const regex archiveURLSignature (".*(([&?]g=)|([&?]l=/)).*"); +static const regex archiveURLRegex (".*(([&?]g=)|([&?]l=/)).*"); static const string KAZ_PLAIN_HR ("______________________________________________________________________________"); static const string KAZ_PLAIN_START ("~~ PJ-KAZ !"); // don't end whith space @@ -423,7 +423,7 @@ MainAttachment::removePreviousArchive () { vector toRemove; for (map ::const_iterator it = previousLinks.begin (); it != previousLinks.end (); ++it) { const string key (it->first); - if (regex_match (key, archiveURLSignature)) + if (regex_match (key, archiveURLRegex)) toRemove.push_back (key); } for (string old : toRemove) @@ -567,7 +567,7 @@ MainAttachment::extract (ifstream &mbox, const SizeArg &minSize) const { // ================================================================================ void -MainAttachment::substitute (ifstream &mbox, ofstream &outbox, const SizeArg &minSize, const AttachMode &attachMode) { +MainAttachment::substitute (ifstream &mbox, ofstream &outbox, const SizeArg &minSize, AttachMode attachMode) { DEF_LOG ("MainAttachment::substitute", "minSize: " << minSize << " AttachMode: " << attachMode); // preparation @@ -601,30 +601,34 @@ MainAttachment::substitute (ifstream &mbox, ofstream &outbox, const SizeArg &min getDisclaim (plainDisclaim, htmlDisclaim); // copy email - if (plainDisclaim.size () && emptyEMail && boundary.empty ()) { - // only one attachment must be replace - cerr << "eMailShrinker: force one attachment" << endl; - string mime (getMime (mbox)); - string::size_type startPos = (0); - for (string token : {string ("Content-Transfer-Encoding"), Attachment::contentTypeToken}) { - startPos = caseInsensitiveFind (mime, "Content-Transfer-Encoding"); - for (string::size_type stopPos (startPos); - (stopPos = mime.find ("\n", stopPos)) != string::npos; - ) { - if (string (" \t").find (mime [stopPos+1]) == string::npos) { - mime.erase (startPos, stopPos-startPos); - break; + if (!boundary.size () && plainDisclaim.size ()) { + if (attachMode & ATTACHMENT) + attachMode = FOOTER; + if (emptyEMail) { + // only one attachment must be replace + cerr << "eMailShrinker: force one attachment" << endl; + string mime (getMime (mbox)); + string::size_type startPos = (0); + for (string token : {string ("Content-Transfer-Encoding"), Attachment::contentTypeToken}) { + startPos = caseInsensitiveFind (mime, "Content-Transfer-Encoding"); + for (string::size_type stopPos (startPos); + (stopPos = mime.find ("\n", stopPos)) != string::npos; + ) { + if (string (" \t").find (mime [stopPos+1]) == string::npos) { + mime.erase (startPos, stopPos-startPos); + break; + } } } + mime.insert (startPos, KAZ_EMPTY_TEXT_PLAIN); + string content (plainDisclaim); + base64Encode (content); + outbox << mime + << content << endl; + outbox.flush (); + outbox.close (); + return; } - mime.insert (startPos, KAZ_EMPTY_TEXT_PLAIN); - string content (plainDisclaim); - base64Encode (content); - outbox << mime - << content << endl; - outbox.flush (); - outbox.close (); - return; } streamoff curPos = 0; copy (mbox, outbox, curPos, contentPos); @@ -694,8 +698,7 @@ MainAttachment::substitute (ifstream &mbox, ofstream &outbox, const SizeArg &min } removeSection (content, KAZ_HTML_START, KAZ_HTML_STOP); removeSection (content, KAZ_PLAIN_START, KAZ_PLAIN_STOP); - // XXX case insensitive ?? - if (content.find (CID) != string::npos) + if (caseInsensitiveFind (content, CID) != string::npos) replaceAll (content, translateHtml); attachP->replaceEmbedded (content); } @@ -733,12 +736,9 @@ MainAttachment::substitute (ifstream &mbox, ofstream &outbox, const SizeArg &min string content (KAZ_HTML_CONTENT+htmlDisclaim+BODY_END+HTML_END); base64Encode (content); - if (boundary.size ()) - outbox << boundary.substr (0, boundary.length () -2) << endl - << KAZ_ATTACHMENT_TEXT_HTML << endl - << content << endl; - else - outbox << "coucou No multipart" << endl; + outbox << boundary.substr (0, boundary.length () -2) << endl + << KAZ_ATTACHMENT_TEXT_HTML << endl + << content << endl; outbox.flush (); } copy (mbox, outbox, curPos, endPos); diff --git a/src/cpp/eMailShrinker.cpp b/src/cpp/eMailShrinker.cpp index 9165e9b..87bad68 100644 --- a/src/cpp/eMailShrinker.cpp +++ b/src/cpp/eMailShrinker.cpp @@ -33,8 +33,8 @@ //////////////////////////////////////////////////////////////////////////// #include "version.hpp" -const std::string kaz::LAST_VERSION_NUM ("2.8"); -const std::string kaz::LAST_VERSION_DATE ("2022-12-23"); +const std::string kaz::LAST_VERSION_NUM ("2.9"); +const std::string kaz::LAST_VERSION_DATE ("2022-12-24"); const std::string kaz::LAST_VERSION (LAST_VERSION_NUM+" "+LAST_VERSION_DATE+" eMailShrinker"); #include @@ -106,7 +106,7 @@ static const char *const inputFileC = inputFile.c_str (); int main (int argc, char** argv) { - // XXX debug before parse options + // uncomment next line in case of debug parse options // Log::debug = true; DEF_LOG ("main:", ""); prog = argv [0]; diff --git a/src/cpp/jirafeauAPI.cpp b/src/cpp/jirafeauAPI.cpp index 46b0706..ec52de8 100644 --- a/src/cpp/jirafeauAPI.cpp +++ b/src/cpp/jirafeauAPI.cpp @@ -106,7 +106,7 @@ static const char *const inputFileC = inputFile.c_str (); int main (int argc, char** argv) { - // XXX debug before parse options + // uncomment next line in case of debug parse options // Log::debug = true; DEF_LOG ("main:", ""); prog = argv [0]; diff --git a/src/cpp/kazMisc.cpp b/src/cpp/kazMisc.cpp index d0cf270..b2907c8 100644 --- a/src/cpp/kazMisc.cpp +++ b/src/cpp/kazMisc.cpp @@ -65,6 +65,14 @@ const string kaz::availableURLChars = "abcdefghijklmnopqrstuvwxyz" "~"; +const regex kaz::encodedWordRegex ("\\s*=\\?" // flag begin + "([0-9A-Za-z!#$%&'+^_`{}~-]+)" // charset + "\\?" // flag sep + "([QqBb])" // quoted our base64 + "\\?" // flag sep + "([^ ?]+)" // encoded string + "\\?=\\s*"); // flag end + // ================================================================================ uint16_t @@ -369,65 +377,66 @@ kaz::iso2utf (string &content) { // ================================================================================ void -kaz::encodedWord (string &content) { +kaz::encodedWordDecode (string &content) { // rfc2047 - DEF_LOG ("kaz::extendedWord", "content: " << content); + DEF_LOG ("kaz::encodedWordDecode", "content: " << content); string::size_type charsetPos = content.find ("=?"); if (charsetPos == string::npos) return; LOG ("charsetPos: " << charsetPos); - LOG_BUG (charsetPos != 0, return, "kazMisc::extendedWord bug: =? not at begin pos. (content: " << content << ")"); string result; - for ( ; - (charsetPos = content.find ("=?", charsetPos)) != string::npos; - ) { - string::size_type modePos = content.find ("?", charsetPos+2); - - LOG_BUG (modePos == string::npos, return, "kazMisc::extendedWord bug: no end chartset. (content: " << content << ")"); - string::size_type contentPos = content.find ("?", modePos+1); + auto pos (0); + sregex_iterator ewItEnd; + for (sregex_iterator ewIt (content.begin (), content.end (), encodedWordRegex); + ewIt != ewItEnd; + ++ewIt) { + smatch m = *ewIt; + if (pos != m.position ()) { + result += content.substr (pos, m.position () - pos); + LOG ("stantad " << content.substr (pos, m.position () - pos)); + } + string encoded (m[3]); + replace (encoded.begin (), encoded.end (), '_', ' '); - LOG_BUG (contentPos != modePos+2, return, "kazMisc::extendedWord bug: no end chartset. (content: " << content << ")"); - string::size_type endPos = content.find ("?=", contentPos+1); + LOG ("charset: " << m[1] << " mode: " << m[2] << " string: " << encoded); - LOG_BUG (endPos == string::npos, return, "kazMisc::extendedWord bug: no end chartset. (content: " << content << ")"); - string tmp (content.substr (contentPos+1, endPos-contentPos-1)); - switch (content [modePos+1]) { + switch (m[2].str ()[0]) { case 'B': case 'b': - base64Decode (tmp); + base64Decode (encoded); break; case 'Q': case 'q': - quotedDecode (tmp); + quotedDecode (encoded); break; default: - LOG_BUG (true, return, "kazMisc::extendedWord bug: unknown mode. (mode: " << content [modePos+1] << ")"); + LOG_BUG (true, return, "kazMisc::encodedWordDecode bug: unknown mode. (mode: " << m[2] << ")"); } - LOG ("tmp: " << tmp); - string charset (content.substr (charsetPos, modePos-charsetPos-2)); + LOG ("decoded: " << encoded); + string charset (m[1]); toLower (charset); if (! caseInsensitiveFind (charset, "ISO")) - iso2utf (tmp); - result += tmp; - charsetPos = endPos+2; + iso2utf (encoded); + result += encoded; + pos = m.position () + m.str ().length (); } - content = result; + content = result + content.substr (pos); LOG ("content: " << content); } // ================================================================================ void -kaz::charsetValue (string &content) { +kaz::charsetValueDecode (string &content) { // rfc2184 - DEF_LOG ("kaz::charsetValue", "content: " << content); + DEF_LOG ("kaz::charsetValueDecode", "content: " << content); string::size_type langPos = content.find ("'"); - LOG_BUG (langPos == string::npos, return, "kazMisc::charsetValue bug: no '. (content: " << content << ")"); + LOG_BUG (langPos == string::npos, return, "kazMisc::charsetValueDecode bug: no '. (content: " << content << ")"); string::size_type contentPos = content.find ("'", langPos+1); - LOG_BUG (contentPos == string::npos, return, "kazMisc::charsetValue bug: no double '. (content: " << content << ")"); + LOG_BUG (contentPos == string::npos, return, "kazMisc::charsetValueDecode bug: no double '. (content: " << content << ")"); string tmp (content.substr (contentPos+1)); quotedDecode<'%'> (tmp); LOG ("tmp: " << tmp); diff --git a/src/include/MainAttachment.hpp b/src/include/MainAttachment.hpp index e093d21..04b16a2 100644 --- a/src/include/MainAttachment.hpp +++ b/src/include/MainAttachment.hpp @@ -126,7 +126,7 @@ namespace kaz { /*! extract big attachments in mbox to extractDir and write to stdout le dirname of each extraction */ void extract (ifstream &mbox, const SizeArg &minSize) const; /*! substitute big attachments by the url give in stdin */ - void substitute (ifstream &mbox, ofstream &outbox, const SizeArg &minSize, const AttachMode &attachMode); + void substitute (ifstream &mbox, ofstream &outbox, const SizeArg &minSize, AttachMode attachMode); }; // ================================================================================ diff --git a/src/include/kazMisc.hpp b/src/include/kazMisc.hpp index 6445024..e636687 100644 --- a/src/include/kazMisc.hpp +++ b/src/include/kazMisc.hpp @@ -38,6 +38,7 @@ #include #include #include +#include namespace kaz { using namespace std; @@ -47,6 +48,8 @@ namespace kaz { extern const char * const base64Chars; /*! set of chars available in URL */ extern const string availableURLChars; + /*! pattern for encoded words */ + extern const regex encodedWordRegex; // ======================================================================= /*! get the width of the terminal */ @@ -82,10 +85,10 @@ namespace kaz { void base64Encode (string &content); /*! side effect to change charset of content */ void iso2utf (string &content); - /*! side effect to get the encodedWord according rfc2047 */ - void encodedWord (string &content); + /*! side effect to get the encoded word according rfc2047 rfc5987 rfc2978 */ + void encodedWordDecode (string &content); /*! side effect to get the charsetValue according rfc2184 */ - void charsetValue (string &content); + void charsetValueDecode (string &content); /*! side effect to remove quote */ void removeQuote (string &content);