//============================================================================= // File: mailbox.cpp // Contents: Definitions for DwMailbox // Maintainer: Doug Sauder <dwsauder@fwb.gulf.net> // WWW: http://www.fwb.gulf.net/~dwsauder/mimepp.html // // Copyright (c) 1996, 1997 Douglas W. Sauder // All rights reserved. // // IN NO EVENT SHALL DOUGLAS W. SAUDER BE LIABLE TO ANY PARTY FOR DIRECT, // INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES ARISING OUT OF // THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF DOUGLAS W. SAUDER // HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // // DOUGLAS W. SAUDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, BUT // NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A // PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS ON AN "AS IS" // BASIS, AND DOUGLAS W. SAUDER HAS NO OBLIGATION TO PROVIDE MAINTENANCE, // SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. // //============================================================================= #define DW_IMPLEMENTATION #include <mimelib/config.h> #include <mimelib/debug.h> #include <ctype.h> #include <stdlib.h> #include <string.h> #include <iostream> #include <mimelib/string.h> #include <mimelib/mailbox.h> #include <mimelib/token.h> void RemoveCrAndLf(DwString& aStr); const char* const DwMailbox::sClassName = "DwMailbox"; DwMailbox* (*DwMailbox::sNewMailbox)(const DwString&, DwMessageComponent*) = 0; DwMailbox* DwMailbox::NewMailbox(const DwString& aStr, DwMessageComponent* aParent) { if (sNewMailbox) { return sNewMailbox(aStr, aParent); } else { return new DwMailbox(aStr, aParent); } } DwMailbox::DwMailbox() { mClassId = kCidMailbox; mClassName = sClassName; } DwMailbox::DwMailbox(const DwMailbox& aMailbox) : DwAddress(aMailbox), mFullName(aMailbox.mFullName), mRoute(aMailbox.mRoute), mLocalPart(aMailbox.mLocalPart), mDomain(aMailbox.mDomain) { mClassId = kCidMailbox; mClassName = sClassName; } DwMailbox::DwMailbox(const DwString& aStr, DwMessageComponent* aParent) : DwAddress(aStr, aParent) { mClassId = kCidMailbox; mClassName = sClassName; } DwMailbox::~DwMailbox() { } const DwMailbox& DwMailbox::operator = (const DwMailbox& aMailbox) { if (this == &aMailbox) return *this; DwAddress::operator = (aMailbox); mFullName = aMailbox.mFullName; mRoute = aMailbox.mRoute; mLocalPart = aMailbox.mLocalPart; mDomain = aMailbox.mDomain; return *this; } const DwString& DwMailbox::FullName() const { return mFullName; } void DwMailbox::SetFullName(const DwString& aFullName) { mFullName = aFullName; SetModified(); } const DwString& DwMailbox::Route() const { return mRoute; } void DwMailbox::SetRoute(const DwString& aRoute) { mRoute = aRoute; SetModified(); } const DwString& DwMailbox::LocalPart() const { return mLocalPart; } void DwMailbox::SetLocalPart(const DwString& aLocalPart) { mLocalPart = aLocalPart; SetModified(); } const DwString& DwMailbox::Domain() const { return mDomain; } void DwMailbox::SetDomain(const DwString& aDomain) { mDomain = aDomain; SetModified(); } // Some mailboxes to test // // John Doe <john.doe@acme.com> // John@acme.com (John Doe) // John.Doe@acme.com (John Doe) // John.Doe (Jr) @acme.com (John Doe) // John <@domain1.com,@domain2.com:jdoe@acme.com> // <jdoe@acme> // <@node1.[128.129.130.131],@node2.uu.edu: // jdoe(John Doe)@node3.[131.130.129.128]> (John Doe) // void DwMailbox::Parse() { mIsModified = 0; DwString emptyString(""); DwString space(" "); int isFirstPhraseNull = 1; int isSimpleAddress = 1; DwString firstPhrase(emptyString); DwString lastComment(emptyString); mRoute = emptyString; mLocalPart = emptyString; mDomain = emptyString; mFullName = emptyString; DwRfc822Tokenizer tokenizer(mString); int ch; enum { eStart, // start eLtSeen, // less-than-seen eInRoute, // in-route eInAddrSpec, // in-addr-spec eAtSeen, // at-seen eGtSeen // greater-than-seen }; // Start state -- terminated by '<' or '@' int type = tokenizer.Type(); int state = eStart; while (state == eStart && type != eTkNull) { switch (type) { case eTkSpecial: ch = tokenizer.Token()[0]; switch (ch) { case '@': state = eAtSeen; break; case '<': isSimpleAddress = 0; mLocalPart = emptyString; state = eLtSeen; break; case '.': mLocalPart += tokenizer.Token(); break; } break; case eTkAtom: case eTkQuotedString: if (isFirstPhraseNull) { firstPhrase = tokenizer.Token(); isFirstPhraseNull = 0; } else { firstPhrase += space; firstPhrase += tokenizer.Token(); } mLocalPart += tokenizer.Token(); break; case eTkComment: tokenizer.StripDelimiters(); lastComment = tokenizer.Token(); break; } ++tokenizer; type = tokenizer.Type(); } // Less-than-seen state -- process only one valid token and transit to // in-route state or in-addr-spec state while (state == eLtSeen && type != eTkNull) { switch (type) { case eTkSpecial: ch = tokenizer.Token()[0]; switch (ch) { case '@': // '@' immediately following '<' indicates a route mRoute = tokenizer.Token(); state = eInRoute; break; } break; case eTkAtom: case eTkQuotedString: mLocalPart = tokenizer.Token(); state = eInAddrSpec; break; } ++tokenizer; type = tokenizer.Type(); } // In-route state -- terminated by ':' while (state == eInRoute && type != eTkNull) { switch (type) { case eTkSpecial: ch = tokenizer.Token()[0]; switch (ch) { case ':': state = eInAddrSpec; break; case '@': case ',': case '.': mRoute += tokenizer.Token(); break; } break; case eTkAtom: mRoute += tokenizer.Token(); break; case eTkDomainLiteral: mRoute += tokenizer.Token(); break; } ++tokenizer; type = tokenizer.Type(); } // in-addr-spec state -- terminated by '@' while (state == eInAddrSpec && type != eTkNull) { switch (type) { case eTkSpecial: ch = tokenizer.Token()[0]; switch (ch) { case '@': state = eAtSeen; break; case '.': mLocalPart += tokenizer.Token(); break; } break; case eTkAtom: case eTkQuotedString: mLocalPart += tokenizer.Token(); break; } ++tokenizer; type = tokenizer.Type(); } // at-seen state -- terminated by '>' or end of string while (state == eAtSeen && type != eTkNull) { switch (type) { case eTkSpecial: ch = tokenizer.Token()[0]; switch (ch) { case '>': state = eGtSeen; break; case '.': mDomain += tokenizer.Token(); break; } break; case eTkAtom: mDomain += tokenizer.Token(); break; case eTkDomainLiteral: mDomain += tokenizer.Token(); break; case eTkComment: tokenizer.StripDelimiters(); lastComment = tokenizer.Token(); break; } ++tokenizer; type = tokenizer.Type(); } // greater-than-seen state -- terminated by end of string while (state == eGtSeen && type != eTkNull) { switch (type) { case eTkComment: tokenizer.StripDelimiters(); lastComment = tokenizer.Token(); break; } ++tokenizer; type = tokenizer.Type(); } // Get full name, if possible if (isSimpleAddress) { mFullName = lastComment; } else if (firstPhrase != emptyString) { mFullName = firstPhrase; } else if (lastComment != emptyString) { mFullName = lastComment; } // Check validity if (mLocalPart.length() > 0) { mIsValid = 1; } else { mIsValid = 0; } // Remove CR or LF from local-part or full name RemoveCrAndLf(mFullName); RemoveCrAndLf(mLocalPart); } void DwMailbox::Assemble() { if (!mIsModified) return; mIsValid = 1; if (mLocalPart.length() == 0 || mDomain.length() == 0) { mIsValid = 0; mString = ""; return; } mString = ""; if (mFullName.length() > 0) { mString += mFullName; mString += " "; } mString += "<"; if (mRoute.length() > 0) { mString += mRoute; mString += ":"; } mString += mLocalPart; mString += "@"; mString += mDomain; mString += ">"; mIsModified = 0; } DwMessageComponent* DwMailbox::Clone() const { return new DwMailbox(*this); } #if defined(DW_DEBUG_VERSION) void DwMailbox::PrintDebugInfo(std::ostream& aStrm, int /*aDepth*/) const { aStrm << "---------------- Debug info for DwMailbox class ----------------\n"; _PrintDebugInfo(aStrm); } #else void DwMailbox::PrintDebugInfo(std::ostream& , int) const {} #endif // defined(DW_DEBUG_VERSION) #if defined(DW_DEBUG_VERSION) void DwMailbox::_PrintDebugInfo(std::ostream& aStrm) const { DwAddress::_PrintDebugInfo(aStrm); aStrm << "Full Name: " << mFullName << '\n'; aStrm << "Route: " << mRoute << '\n'; aStrm << "Local Part: " << mLocalPart << '\n'; aStrm << "Domain: " << mDomain << '\n'; } #else void DwMailbox::_PrintDebugInfo(std::ostream& ) const {} #endif // defined(DW_DEBUG_VERSION) void DwMailbox::CheckInvariants() const { #if defined(DW_DEBUG_VERSION) DwAddress::CheckInvariants(); mFullName.CheckInvariants(); mRoute.CheckInvariants(); mLocalPart.CheckInvariants(); mDomain.CheckInvariants(); #endif // defined(DW_DEBUG_VERSION) } void RemoveCrAndLf(DwString& aStr) { // Do a quick check to see if at least one CR or LF is present size_t n = aStr.find_first_of("\r\n"); if (n == DwString::npos) return; // At least one CR or LF is present, so copy the string const DwString& in = aStr; size_t inLen = in.length(); DwString out; out.reserve(inLen); int lastChar = 0; size_t i = 0; while (i < inLen) { int ch = in[i]; if (ch == (int) '\r') { out += ' '; } else if (ch == (int) '\n') { if (lastChar != (int) '\r') { out += ' '; } } else { out += (char) ch; } lastChar = ch; ++i; } aStr = out; }