diff options
Diffstat (limited to 'mimelib/headers.cpp')
-rw-r--r-- | mimelib/headers.cpp | 1035 |
1 files changed, 1035 insertions, 0 deletions
diff --git a/mimelib/headers.cpp b/mimelib/headers.cpp new file mode 100644 index 000000000..7598ec7ea --- /dev/null +++ b/mimelib/headers.cpp @@ -0,0 +1,1035 @@ +//============================================================================= +// File: headers.cpp +// Contents: Definitions for DwHeaders +// 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 <time.h> +#include <iostream> +#include <mimelib/string.h> +#include <mimelib/headers.h> +#include <mimelib/field.h> +#include <mimelib/body.h> +#include <mimelib/datetime.h> +#include <mimelib/mailbox.h> +#include <mimelib/address.h> +#include <mimelib/mechansm.h> +#include <mimelib/mediatyp.h> +#include <mimelib/msgid.h> +#include <mimelib/text.h> + + +class DwHeadersParser { + friend class DwHeaders; +private: + DwHeadersParser(const DwString&); + void Rewind(); + void NextField(DwString*); + const DwString mString; + size_t mPos; +}; + + +DwHeadersParser::DwHeadersParser(const DwString& aStr) + : mString(aStr) +{ + mPos = 0; +} + + +void DwHeadersParser::Rewind() +{ + mPos = 0; +} + + +void DwHeadersParser::NextField(DwString* aStr) +{ + if (!aStr) { + return; + } + const char* buf = mString.data(); + size_t bufEnd = mString.length(); + size_t pos = mPos; + size_t start = pos; + size_t len = 0; + while (pos < bufEnd) { + if (buf[pos] == '\n' + && pos+1 < bufEnd + && buf[pos+1] != ' ' + && buf[pos+1] != '\t') { + + ++len; + ++pos; + break; + } + ++len; + ++pos; + } + *aStr = mString.substr(start, len); + mPos = pos; +} + + +//============================================================================ + + +const char* const DwHeaders::sClassName = "DwHeaders"; + + +DwHeaders* (*DwHeaders::sNewHeaders)(const DwString&, DwMessageComponent*) = 0; + + +DwHeaders* DwHeaders::NewHeaders(const DwString& aStr, + DwMessageComponent* aParent) +{ + if (sNewHeaders) { + return sNewHeaders(aStr, aParent); + } + else { + return new DwHeaders(aStr, aParent); + } +} + + +DwHeaders::DwHeaders() +{ + mFirstField = 0; + mLastField = 0; + mClassId = kCidHeaders; + mClassName = sClassName; +} + + +DwHeaders::DwHeaders(const DwHeaders& aHeader) + : DwMessageComponent(aHeader) +{ + mFirstField = 0; + mLastField = 0; + if (aHeader.mFirstField) { + CopyFields(aHeader.mFirstField); + } + mClassId = kCidHeaders; + mClassName = sClassName; +} + + +DwHeaders::DwHeaders(const DwString& aStr, DwMessageComponent* aParent) + : DwMessageComponent(aStr, aParent) +{ + mFirstField = 0; + mLastField = 0; + mClassId = kCidHeaders; + mClassName = sClassName; +} + + +DwHeaders::~DwHeaders() +{ + if (mFirstField) { + DeleteAllFields(); + } +} + + +const DwHeaders& DwHeaders::operator = (const DwHeaders& aHeader) +{ + if (this == &aHeader) return *this; + DwMessageComponent::operator = (aHeader); + if (mFirstField) { + DeleteAllFields(); + } + if (aHeader.mFirstField) { + CopyFields(aHeader.mFirstField); + } + if (mParent) { + mParent->SetModified(); + } + return *this; +} + + +void DwHeaders::Parse() +{ + mIsModified = 0; + DwHeadersParser parser(mString); + DwString str; + parser.NextField(&str); + while (!str.empty()) { + DwField* field = DwField::NewField(str, this); + field->Parse(); + _AddField(field); + parser.NextField(&str); + } +} + + +void DwHeaders::Assemble() +{ + if (!mIsModified) return; + mString = ""; + DwField* field = FirstField(); + while (field) { + field->Assemble(); + mString += field->AsString(); + field = field->Next(); + } + // We DwEntityParser skips the empty line separating the headers + // from the body, so why should be add it here? + //mString += DW_EOL; + mIsModified = 0; +} + + +DwMessageComponent* DwHeaders::Clone() const +{ + return new DwHeaders(*this); +} + + +DwFieldBody& DwHeaders::FieldBody(const DwString& aFieldName) +{ + assert(!aFieldName.empty()); + // First, search for field + DwField* field = FindField(aFieldName); + // If the field is not found, create the field and its field body + if (field == 0) { + field = DwField::NewField("", this); + field->SetFieldNameStr(aFieldName); + DwFieldBody* fieldBody = DwField::CreateFieldBody(aFieldName, + "", field); + field->SetFieldBody(fieldBody); + AddField(field); + } + // Get the field body + DwFieldBody* fieldBody = field->FieldBody(); + // If it does not exist, create it + if (fieldBody == 0) { + fieldBody = DwField::CreateFieldBody(aFieldName, "", field); + field->SetFieldBody(fieldBody); + SetModified(); + } + return *fieldBody; +} + + +std::vector<DwFieldBody*> DwHeaders::AllFieldBodies(const DwString& aFieldName) +{ + assert(!aFieldName.empty()); + // First, search for field + DwField* field = FindField(aFieldName); + // If the field is not found, create the field and its field body + if (field == 0) { + field = DwField::NewField("", this); + field->SetFieldNameStr(aFieldName); + DwFieldBody* fieldBody = DwField::CreateFieldBody(aFieldName, + "", field); + field->SetFieldBody(fieldBody); + AddField(field); + } + std::vector<DwFieldBody*> v; + for ( ; field; field = field->Next() ) { + if (DwStrcasecmp(field->FieldNameStr(), aFieldName) == 0) { + // Get the field body + DwFieldBody* fieldBody = field->FieldBody(); + // If it does not exist, create it + if (fieldBody == 0) { + fieldBody = DwField::CreateFieldBody(aFieldName, "", field); + field->SetFieldBody(fieldBody); + SetModified(); + } + v.push_back( fieldBody ); + } + } + return v; +} + + +int DwHeaders::NumFields() const +{ + int count = 0; + DwField* field = mFirstField; + while (field) { + ++count; + field = field->Next(); + } + return count; +} + + +DwField* DwHeaders::FindField(const char* aFieldName) const +{ + assert(aFieldName != 0); + if (aFieldName == 0) return 0; + DwField* field = mFirstField; + while (field) { + if (DwStrcasecmp(field->FieldNameStr(), aFieldName) == 0) { + break; + } + field = field->Next(); + } + return field; +} + + +DwField* DwHeaders::FindField(const DwString& aFieldName) const +{ + DwField* field = mFirstField; + while (field) { + if (DwStrcasecmp(field->FieldNameStr(), aFieldName) == 0) { + break; + } + field = field->Next(); + } + return field; +} + + +void DwHeaders::AddOrReplaceField(DwField* aField) +{ + assert(aField != 0); + if (aField == 0) return; + SetModified(); + const DwString& fieldName = aField->FieldNameStr(); + DwField* prevField = 0; + DwField* field = mFirstField; + while (field) { + if (DwStrcasecmp(field->FieldNameStr(), fieldName) == 0) { + break; + } + prevField = field; + field = field->Next(); + } + // Field was not found, so just add it + if (!field) { + _AddField(aField); + } + // Field was found. Replace the old one with the new one. + else { + if (prevField) { + prevField->SetNext(aField); + } + else { + mFirstField = aField; + } + aField->SetNext(field->Next()); + // Check whether we've replaced the last field + if ( !aField->Next() ) + mLastField = aField; + delete field; + } +} + + +void DwHeaders::AddField(DwField* aField) +{ + assert(aField != 0); + if (aField == 0) return; + _AddField(aField); + SetModified(); +} + + +void DwHeaders::AddFieldAt(int aPos, DwField* aField) +{ + assert(aField != 0); + if (aField == 0) return; + SetModified(); + // Special case: empty list + if (mFirstField == 0) { + aField->SetNext(0); + mFirstField = aField; + mLastField = aField; + return; + } + // Special case: aPos == 1 --> add at beginning + if (aPos == 1) { + aField->SetNext(mFirstField); + mFirstField = aField; + return; + } + // aPos == 0 --> at at end + if (aPos == 0) { + _AddField(aField); + return; + } + int count = 2; + DwField* field = mFirstField; + while (field->Next() && count < aPos) { + field = field->Next(); + ++count; + } + aField->SetNext(field->Next()); + field->SetNext(aField); + // Check whether we've a new last field + if ( !aField->Next() ) + mLastField = aField; +} + + +void DwHeaders::RemoveField(DwField* aField) +{ + DwField* prevField = 0; + DwField* field = mFirstField; + while (field) { + if (field == aField) { + break; + } + prevField = field; + field = field->Next(); + } + // If we found the field... + if (field) { + if (prevField == 0) { + mFirstField = field->Next(); + } + else { + prevField->SetNext(field->Next()); + } + // Check whether we've removed the last field + if ( field == mLastField ) + mLastField = prevField; + field->SetNext(0); + SetModified(); + } +} + + +void DwHeaders::DeleteAllFields() +{ + DwField* field = mFirstField; + while (field) { + DwField* nextField = field->Next(); + delete field; + field = nextField; + } + mFirstField = 0; + mLastField = 0; +} + + +void DwHeaders::_AddField(DwField* aField) +{ + if (aField == 0) return; + // Add field with setting is-modified flag for header + aField->SetParent(this); + // Special case: empty list + if (mFirstField == 0) { + mFirstField = aField; + mLastField = aField; + return; + } + mLastField->SetNext(aField); + mLastField = aField; +} + + +void DwHeaders::CopyFields(DwField* aFirst) +{ + DwField* field = aFirst; + DwField* newField; + while (field) { + newField = (DwField*) field->Clone(); + _AddField(newField); + field = field->Next(); + } +} + + +DwBool DwHeaders::HasBcc() const +{ + return FindField("bcc") ? 1 : 0; +} + + +DwBool DwHeaders::HasCc() const +{ + return FindField("cc") ? 1 : 0; +} + + +DwBool DwHeaders::HasComments() const +{ + return FindField("comments") ? 1 : 0; +} + + +DwBool DwHeaders::HasDate() const +{ + return FindField("date") ? 1 : 0; +} + + +DwBool DwHeaders::HasEncrypted() const +{ + return FindField("encrypted") ? 1 : 0; +} + + +DwBool DwHeaders::HasFrom() const +{ + return FindField("from") ? 1 : 0; +} + + +DwBool DwHeaders::HasInReplyTo() const +{ + return FindField("in-reply-to") ? 1 : 0; +} + + +DwBool DwHeaders::HasKeywords() const +{ + return FindField("keywords") ? 1 : 0; +} + + +DwBool DwHeaders::HasMessageId() const +{ + return FindField("message-id") ? 1 : 0; +} + + +DwBool DwHeaders::HasReceived() const +{ + return FindField("received") ? 1 : 0; +} + + +DwBool DwHeaders::HasReferences() const +{ + return FindField("references") ? 1 : 0; +} + + +DwBool DwHeaders::HasReplyTo() const +{ + return FindField("reply-to") ? 1 : 0; +} + + +DwBool DwHeaders::HasResentBcc() const +{ + return FindField("resent-bcc") ? 1 : 0; +} + + +DwBool DwHeaders::HasResentCc() const +{ + return FindField("resent-cc") ? 1 : 0; +} + + +DwBool DwHeaders::HasResentDate() const +{ + return FindField("resent-date") ? 1 : 0; +} + + +DwBool DwHeaders::HasResentFrom() const +{ + return FindField("resent-from") ? 1 : 0; +} + + +DwBool DwHeaders::HasResentMessageId() const +{ + return FindField("resent-message-id") ? 1 : 0; +} + + +DwBool DwHeaders::HasResentReplyTo() const +{ + return FindField("resent-reply-to") ? 1 : 0; +} + + +DwBool DwHeaders::HasResentSender() const +{ + return FindField("resent-sender") ? 1 : 0; +} + + +DwBool DwHeaders::HasResentTo() const +{ + return FindField("resent-to") ? 1 : 0; +} + + +DwBool DwHeaders::HasReturnPath() const +{ + return FindField("return-path") ? 1 : 0; +} + + +DwBool DwHeaders::HasSender() const +{ + return FindField("sender") ? 1 : 0; +} + + +DwBool DwHeaders::HasSubject() const +{ + return FindField("subject") ? 1 : 0; +} + + +DwBool DwHeaders::HasTo() const +{ + return FindField("to") ? 1 : 0; +} + + +DwBool DwHeaders::HasApproved() const +{ + return FindField("approved") ? 1 : 0; +} + + +DwBool DwHeaders::HasControl() const +{ + return FindField("control") ? 1 : 0; +} + + +DwBool DwHeaders::HasDistribution() const +{ + return FindField("distribution") ? 1 : 0; +} + + +DwBool DwHeaders::HasExpires() const +{ + return FindField("expires") ? 1 : 0; +} + + +DwBool DwHeaders::HasFollowupTo() const +{ + return FindField("followup-to") ? 1 : 0; +} + + +DwBool DwHeaders::HasLines() const +{ + return FindField("lines") ? 1 : 0; +} + + +DwBool DwHeaders::HasNewsgroups() const +{ + return FindField("newsgroups") ? 1 : 0; +} + + +DwBool DwHeaders::HasOrganization() const +{ + return FindField("organization") ? 1 : 0; +} + + +DwBool DwHeaders::HasPath() const +{ + return FindField("path") ? 1 : 0; +} + + +DwBool DwHeaders::HasSummary() const +{ + return FindField("summary") ? 1 : 0; +} + + +DwBool DwHeaders::HasXref() const +{ + return FindField("xref") ? 1 : 0; +} + + +DwBool DwHeaders::HasContentDescription() const +{ + return FindField("content-description") ? 1 : 0; +} + + +DwBool DwHeaders::HasContentId() const +{ + return FindField("content-id") ? 1 : 0; +} + + +DwBool DwHeaders::HasContentTransferEncoding() const +{ + return FindField("content-transfer-encoding") ? 1 : 0; +} + + +DwBool DwHeaders::HasCte() const +{ + return FindField("content-transfer-encoding") ? 1 : 0; +} + + +DwBool DwHeaders::HasContentType() const +{ + return FindField("content-type") ? 1 : 0; +} + + +DwBool DwHeaders::HasMimeVersion() const +{ + return FindField("mime-version") ? 1 : 0; +} + + +DwBool DwHeaders::HasContentDisposition() const +{ + return FindField("content-disposition") ? 1 : 0; +} + + +DwBool DwHeaders::HasField(const char* aFieldName) const +{ + return FindField(aFieldName) ? 1 : 0; +} + + +DwBool DwHeaders::HasField(const DwString& aFieldName) const +{ + return FindField(aFieldName) ? 1 : 0; +} + + +DwAddressList& DwHeaders::Bcc() +{ + return (DwAddressList&) FieldBody("Bcc"); +} + + +DwAddressList& DwHeaders::Cc() +{ + return (DwAddressList&) FieldBody("Cc"); +} + + +DwText& DwHeaders::Comments() +{ + return (DwText&) FieldBody("Comments"); +} + + +DwDateTime& DwHeaders::Date() +{ + return (DwDateTime&) FieldBody("Date"); +} + + +DwText& DwHeaders::Encrypted() +{ + return (DwText&) FieldBody("Encrypted"); +} + + +DwMailboxList& DwHeaders::From() +{ + return (DwMailboxList&) FieldBody("From"); +} + + +DwText& DwHeaders::InReplyTo() +{ + return (DwText&) FieldBody("In-Reply-To"); +} + + +DwText& DwHeaders::Keywords() +{ + return (DwText&) FieldBody("Keywords"); +} + + +DwMsgId& DwHeaders::MessageId() +{ + return (DwMsgId&) FieldBody("Message-Id"); +} + + +DwText& DwHeaders::Received() +{ + return (DwText&) FieldBody("Received"); +} + + +DwText& DwHeaders::References() +{ + return (DwText&) FieldBody("References"); +} + + +DwAddressList& DwHeaders::ReplyTo() +{ + return (DwAddressList&) FieldBody("Reply-To"); +} + + +DwAddressList& DwHeaders::ResentBcc() +{ + return (DwAddressList&) FieldBody("Resent-Bcc"); +} + + +DwAddressList& DwHeaders::ResentCc() +{ + return (DwAddressList&) FieldBody("Resent-Cc"); +} + + +DwDateTime& DwHeaders::ResentDate() +{ + return (DwDateTime&) FieldBody("Resent-Date"); +} + + +DwMailboxList& DwHeaders::ResentFrom() +{ + return (DwMailboxList&) FieldBody("Resent-From"); +} + + +DwMsgId& DwHeaders::ResentMessageId() +{ + return (DwMsgId&) FieldBody("Resent-Message-Id"); +} + + +DwAddressList& DwHeaders::ResentReplyTo() +{ + return (DwAddressList&) FieldBody("Resent-Reply-To"); +} + + +DwMailbox& DwHeaders::ResentSender() +{ + return (DwMailbox&) FieldBody("Resent-Sender"); +} + + +DwAddressList& DwHeaders::ResentTo() +{ + return (DwAddressList&) FieldBody("Resent-To"); +} + + +DwAddress& DwHeaders::ReturnPath() +{ + return (DwAddress&) FieldBody("Return-Path"); +} + + +DwMailbox& DwHeaders::Sender() +{ + return (DwMailbox&) FieldBody("Sender"); +} + + +DwText& DwHeaders::Subject() +{ + return (DwText&) FieldBody("Subject"); +} + + +DwAddressList& DwHeaders::To() +{ + return (DwAddressList&) FieldBody("To"); +} + + +DwText& DwHeaders::Approved() +{ + return (DwText&) FieldBody("Approved"); +} + + +DwText& DwHeaders::Control() +{ + return (DwText&) FieldBody("Control"); +} + + +DwText& DwHeaders::Distribution() +{ + return (DwText&) FieldBody("Distribution"); +} + + +DwText& DwHeaders::Expires() +{ + return (DwText&) FieldBody("Expires"); +} + + +DwText& DwHeaders::FollowupTo() +{ + return (DwText&) FieldBody("Followup-To"); +} + + +DwText& DwHeaders::Lines() +{ + return (DwText&) FieldBody("Lines"); +} + + +DwText& DwHeaders::Newsgroups() +{ + return (DwText&) FieldBody("Newsgroups"); +} + + +DwText& DwHeaders::Organization() +{ + return (DwText&) FieldBody("Organization"); +} + + +DwText& DwHeaders::Path() +{ + return (DwText&) FieldBody("Path"); +} + + +DwText& DwHeaders::Summary() +{ + return (DwText&) FieldBody("Summary"); +} + + +DwText& DwHeaders::Xref() +{ + return (DwText&) FieldBody("Xref"); +} + + + +DwText& DwHeaders::ContentDescription() +{ + return (DwText&) FieldBody("Content-Description"); +} + + +DwMsgId& DwHeaders::ContentId() +{ + return (DwMsgId&) FieldBody("Content-Id"); +} + + +DwMechanism& DwHeaders::ContentTransferEncoding() +{ + return (DwMechanism&) + FieldBody("Content-Transfer-Encoding"); +} + + +DwMechanism& DwHeaders::Cte() +{ + return (DwMechanism&) + FieldBody("Content-Transfer-Encoding"); +} + + +DwMediaType& DwHeaders::ContentType() +{ + return (DwMediaType&) FieldBody("Content-Type"); +} + + +DwText& DwHeaders::MimeVersion() +{ + return (DwText&) FieldBody("MIME-Version"); +} + + +DwDispositionType& DwHeaders::ContentDisposition() +{ + return (DwDispositionType&) FieldBody("Content-Disposition"); +} + + +#if defined (DW_DEBUG_VERSION) +void DwHeaders::PrintDebugInfo(std::ostream& aStrm, int aDepth) const +{ + aStrm << + "---------------- Debug info for DwHeaders class ----------------\n"; + _PrintDebugInfo(aStrm); + int depth = aDepth - 1; + depth = (depth >= 0) ? depth : 0; + if (aDepth == 0 || depth > 0) { + DwField* field = mFirstField; + while (field) { + field->PrintDebugInfo(aStrm, depth); + field = (DwField*) field->Next(); + } + } +} +#else +void DwHeaders::PrintDebugInfo(std::ostream& , int ) const {} +#endif // defined (DW_DEBUG_VERSION) + + +#if defined (DW_DEBUG_VERSION) +void DwHeaders::_PrintDebugInfo(std::ostream& aStrm) const +{ + DwMessageComponent::_PrintDebugInfo(aStrm); + aStrm << "Fields: "; + int count = 0; + DwField* field = mFirstField; + while (field) { + if (count > 0) aStrm << ' '; + aStrm << field->ObjectId(); + field = (DwField*) field->Next(); + ++count; + } + aStrm << '\n'; +} +#else +void DwHeaders::_PrintDebugInfo(std::ostream& ) const {} +#endif // defined (DW_DEBUG_VERSION) + + +void DwHeaders::CheckInvariants() const +{ +#if defined (DW_DEBUG_VERSION) + DwMessageComponent::CheckInvariants(); + DwField* field = mFirstField; + while (field) { + field->CheckInvariants(); + assert((DwMessageComponent*) this == field->Parent()); + field = (DwField*) field->Next(); + } +#endif // defined (DW_DEBUG_VERSION) +} + + |