/// <summary> /// Returns address as string value. /// </summary> /// <param name="wordEncoder">8-bit words ecnoder. Value null means that words are not encoded.</param> /// <returns>Returns address as string value.</returns> public override string ToString(MIME_Encoding_EncodedWord wordEncoder) { StringBuilder retVal = new StringBuilder(); if(string.IsNullOrEmpty(m_DisplayName)){ retVal.Append(":"); } else{ if(MIME_Encoding_EncodedWord.MustEncode(m_DisplayName)){ retVal.Append(wordEncoder.Encode(m_DisplayName) + ":"); } else{ retVal.Append(TextUtils.QuoteString(m_DisplayName) + ":"); } } for(int i=0;i<m_pList.Count;i++){ retVal.Append(m_pList[i].ToString(wordEncoder)); if(i < (m_pList.Count - 1)){ retVal.Append(","); } } retVal.Append(";"); return retVal.ToString(); }
/// <summary> /// Constructs ENVELOPE address structure. /// </summary> /// <param name="address">Mailbox address.</param> /// <param name="wordEncoder">Unicode words encoder.</param> /// <returns></returns> private static string ConstructAddress(Mail_t_Mailbox address,MIME_Encoding_EncodedWord wordEncoder) { /* An address structure is a parenthesized list that describes an electronic mail address. The fields of an address structure are in the following order: personal name, [SMTP] at-domain-list (source route), mailbox name, and host name. */ // NOTE: all header fields and parameters must in ENCODED form !!! StringBuilder retVal = new StringBuilder(); retVal.Append("("); // personal name if(address.DisplayName != null){ retVal.Append(TextUtils.QuoteString(wordEncoder.Encode(RemoveCrlf(address.DisplayName)))); } else{ retVal.Append("NIL"); } // source route, always NIL (not used nowdays) retVal.Append(" NIL"); // mailbox name retVal.Append(" " + TextUtils.QuoteString(wordEncoder.Encode(RemoveCrlf(address.LocalPart)))); // host name if(address.Domain != null){ retVal.Append(" " + TextUtils.QuoteString(wordEncoder.Encode(RemoveCrlf(address.Domain)))); } else{ retVal.Append(" NIL"); } retVal.Append(")"); return retVal.ToString(); }
/// <summary> /// Construct secified mime entity ENVELOPE string. /// </summary> /// <param name="entity">Mail message.</param> /// <returns></returns> public static string ConstructEnvelope(Mail_Message entity) { /* RFC 3501 7.4.2 ENVELOPE A parenthesized list that describes the envelope structure of a message. This is computed by the server by parsing the [RFC-2822] header into the component parts, defaulting various fields as necessary. The fields of the envelope structure are in the following order: date, subject, from, sender, reply-to, to, cc, bcc, in-reply-to, and message-id. The date, subject, in-reply-to, and message-id fields are strings. The from, sender, reply-to, to, cc, and bcc fields are parenthesized lists of address structures. An address structure is a parenthesized list that describes an electronic mail address. The fields of an address structure are in the following order: personal name, [SMTP] at-domain-list (source route), mailbox name, and host name. [RFC-2822] group syntax is indicated by a special form of address structure in which the host name field is NIL. If the mailbox name field is also NIL, this is an end of group marker (semi-colon in RFC 822 syntax). If the mailbox name field is non-NIL, this is a start of group marker, and the mailbox name field holds the group name phrase. If the Date, Subject, In-Reply-To, and Message-ID header lines are absent in the [RFC-2822] header, the corresponding member of the envelope is NIL; if these header lines are present but empty the corresponding member of the envelope is the empty string. Note: some servers may return a NIL envelope member in the "present but empty" case. Clients SHOULD treat NIL and empty string as identical. Note: [RFC-2822] requires that all messages have a valid Date header. Therefore, the date member in the envelope can not be NIL or the empty string. Note: [RFC-2822] requires that the In-Reply-To and Message-ID headers, if present, have non-empty content. Therefore, the in-reply-to and message-id members in the envelope can not be the empty string. If the From, To, cc, and bcc header lines are absent in the [RFC-2822] header, or are present but empty, the corresponding member of the envelope is NIL. If the Sender or Reply-To lines are absent in the [RFC-2822] header, or are present but empty, the server sets the corresponding member of the envelope to be the same value as the from member (the client is not expected to know to do this). Note: [RFC-2822] requires that all messages have a valid From header. Therefore, the from, sender, and reply-to members in the envelope can not be NIL. ENVELOPE ("date" "subject" from sender reply-to to cc bcc "in-reply-to" "messageID") */ // NOTE: all header fields and parameters must in ENCODED form !!! MIME_Encoding_EncodedWord wordEncoder = new MIME_Encoding_EncodedWord(MIME_EncodedWordEncoding.B,Encoding.UTF8); wordEncoder.Split = false; StringBuilder retVal = new StringBuilder(); retVal.Append("ENVELOPE ("); // date try{ if(entity.Date != DateTime.MinValue){ retVal.Append(TextUtils.QuoteString(MIME_Utils.DateTimeToRfc2822(entity.Date))); } else{ retVal.Append("NIL"); } } catch{ retVal.Append("NIL"); } // subject if(entity.Subject != null){ //retVal.Append(" " + TextUtils.QuoteString(wordEncoder.Encode(entity.Subject))); string val = wordEncoder.Encode(entity.Subject); retVal.Append(" {" + val.Length + "}\r\n" + val); } else{ retVal.Append(" NIL"); } // from if(entity.From != null && entity.From.Count > 0){ retVal.Append(" " + ConstructAddresses(entity.From.ToArray(),wordEncoder)); } else{ retVal.Append(" NIL"); } // sender // NOTE: There is confusing part, according rfc 2822 Sender: is MailboxAddress and not AddressList. if(entity.Sender != null){ retVal.Append(" ("); retVal.Append(ConstructAddress(entity.Sender,wordEncoder)); retVal.Append(")"); } else{ retVal.Append(" NIL"); } // reply-to if(entity.ReplyTo != null){ retVal.Append(" " + ConstructAddresses(entity.ReplyTo.Mailboxes,wordEncoder)); } else{ retVal.Append(" NIL"); } // to if(entity.To != null && entity.To.Count > 0){ retVal.Append(" " + ConstructAddresses(entity.To.Mailboxes,wordEncoder)); } else{ retVal.Append(" NIL"); } // cc if(entity.Cc != null && entity.Cc.Count > 0){ retVal.Append(" " + ConstructAddresses(entity.Cc.Mailboxes,wordEncoder)); } else{ retVal.Append(" NIL"); } // bcc if(entity.Bcc != null && entity.Bcc.Count > 0){ retVal.Append(" " + ConstructAddresses(entity.Bcc.Mailboxes,wordEncoder)); } else{ retVal.Append(" NIL"); } // in-reply-to if(entity.InReplyTo != null){ retVal.Append(" " + TextUtils.QuoteString(wordEncoder.Encode(entity.InReplyTo))); } else{ retVal.Append(" NIL"); } // message-id if(entity.MessageID != null){ retVal.Append(" " + TextUtils.QuoteString(wordEncoder.Encode(entity.MessageID))); } else{ retVal.Append(" NIL"); } retVal.Append(")"); return retVal.ToString(); }
/// <summary> /// Constructs specified entity and it's childentities bodystructure string. /// </summary> /// <param name="entity">Mime entity.</param> /// <param name="bodystructure">Specifies if to construct BODY or BODYSTRUCTURE.</param> /// <returns></returns> private static string ConstructParts(MIME_Entity entity,bool bodystructure) { /* RFC 3501 7.4.2 BODYSTRUCTURE BODY A form of BODYSTRUCTURE without extension data. A parenthesized list that describes the [MIME-IMB] body structure of a message. This is computed by the server by parsing the [MIME-IMB] header fields, defaulting various fields as necessary. For example, a simple text message of 48 lines and 2279 octets can have a body structure of: ("TEXT" "PLAIN" ("CHARSET" "US-ASCII") NIL NIL "7BIT" 2279 48) Multiple parts are indicated by parenthesis nesting. Instead of a body type as the first element of the parenthesized list, there is a sequence of one or more nested body structures. The second element of the parenthesized list is the multipart subtype (mixed, digest, parallel, alternative, etc.). For example, a two part message consisting of a text and a BASE64-encoded text attachment can have a body structure of: (("TEXT" "PLAIN" ("CHARSET" "US-ASCII") NIL NIL "7BIT" 1152 23)("TEXT" "PLAIN" ("CHARSET" "US-ASCII" "NAME" "cc.diff") "<*****@*****.**>" "Compiler diff" "BASE64" 4554 73) "MIXED") Extension data follows the multipart subtype. Extension data is never returned with the BODY fetch, but can be returned with a BODYSTRUCTURE fetch. Extension data, if present, MUST be in the defined order. The extension data of a multipart body part are in the following order: body parameter parenthesized list A parenthesized list of attribute/value pairs [e.g., ("foo" "bar" "baz" "rag") where "bar" is the value of "foo", and "rag" is the value of "baz"] as defined in [MIME-IMB]. body disposition A parenthesized list, consisting of a disposition type string, followed by a parenthesized list of disposition attribute/value pairs as defined in [DISPOSITION]. body language A string or parenthesized list giving the body language value as defined in [LANGUAGE-TAGS]. body location A string list giving the body content URI as defined in [LOCATION]. Any following extension data are not yet defined in this version of the protocol. Such extension data can consist of zero or more NILs, strings, numbers, or potentially nested parenthesized lists of such data. Client implementations that do a BODYSTRUCTURE fetch MUST be prepared to accept such extension data. Server implementations MUST NOT send such extension data until it has been defined by a revision of this protocol. The basic fields of a non-multipart body part are in the following order: body type A string giving the content media type name as defined in [MIME-IMB]. body subtype A string giving the content subtype name as defined in [MIME-IMB]. body parameter parenthesized list A parenthesized list of attribute/value pairs [e.g., ("foo" "bar" "baz" "rag") where "bar" is the value of "foo" and "rag" is the value of "baz"] as defined in [MIME-IMB]. body id A string giving the content id as defined in [MIME-IMB]. body description A string giving the content description as defined in [MIME-IMB]. body encoding A string giving the content transfer encoding as defined in [MIME-IMB]. body size A number giving the size of the body in octets. Note that this size is the size in its transfer encoding and not the resulting size after any decoding. A body type of type MESSAGE and subtype RFC822 contains, immediately after the basic fields, the envelope structure, body structure, and size in text lines of the encapsulated message. A body type of type TEXT contains, immediately after the basic fields, the size of the body in text lines. Note that this size is the size in its content transfer encoding and not the resulting size after any decoding. Extension data follows the basic fields and the type-specific fields listed above. Extension data is never returned with the BODY fetch, but can be returned with a BODYSTRUCTURE fetch. Extension data, if present, MUST be in the defined order. The extension data of a non-multipart body part are in the following order: body MD5 A string giving the body MD5 value as defined in [MD5]. body disposition A parenthesized list with the same content and function as the body disposition for a multipart body part. body language A string or parenthesized list giving the body language value as defined in [LANGUAGE-TAGS]. body location A string list giving the body content URI as defined in [LOCATION]. Any following extension data are not yet defined in this version of the protocol, and would be as described above under multipart extension data. // We don't construct extention fields like rfc says: Server implementations MUST NOT send such extension data until it has been defined by a revision of this protocol. contentTypeMainMediaType - Example: 'TEXT' contentTypeSubMediaType - Example: 'PLAIN' conentTypeParameters - Example: '("CHARSET" "iso-8859-1" ...)' contentID - Content-ID: header field value. contentDescription - Content-Description: header field value. contentEncoding - Content-Transfer-Encoding: header field value. contentSize - mimeEntity ENCODED data size [envelope] - NOTE: included only if contentType = "message" !!! [contentLines] - number of ENCODED data lines. NOTE: included only if contentType = "text" !!! // Basic fields for multipart (nestedMimeEntries) contentTypeSubMediaType // Basic fields for non-multipart contentTypeMainMediaType contentTypeSubMediaType (conentTypeParameters) contentID contentDescription contentEncoding contentSize [envelope] [contentLine] */ MIME_Encoding_EncodedWord wordEncoder = new MIME_Encoding_EncodedWord(MIME_EncodedWordEncoding.B,Encoding.UTF8); wordEncoder.Split = false; StringBuilder retVal = new StringBuilder(); // Multipart message if(entity.Body is MIME_b_Multipart){ retVal.Append("("); // Construct child entities. foreach(MIME_Entity childEntity in ((MIME_b_Multipart)entity.Body).BodyParts){ // Construct child entity. This can be multipart or non multipart. retVal.Append(ConstructParts(childEntity,bodystructure)); } // Add contentTypeSubMediaType if(entity.ContentType != null && entity.ContentType.SubType != null){ retVal.Append(" \"" + entity.ContentType.SubType + "\""); } else{ retVal.Append(" NIL"); } retVal.Append(")"); } // Single part message else{ retVal.Append("("); // NOTE: all header fields and parameters must in ENCODED form !!! // Add contentTypeMainMediaType if(entity.ContentType != null && entity.ContentType.Type != null){ retVal.Append("\"" + entity.ContentType.Type + "\""); } else{ retVal.Append("NIL"); } // Add contentTypeSubMediaType if(entity.ContentType != null && entity.ContentType.SubType != null){ retVal.Append(" \"" + entity.ContentType.SubType + "\""); } else{ retVal.Append(" NIL"); } // conentTypeParameters - Syntax: {("name" SP "value" *(SP "name" SP "value"))} if(entity.ContentType != null){ if(entity.ContentType.Parameters.Count > 0){ retVal.Append(" ("); bool first = true; foreach(MIME_h_Parameter parameter in entity.ContentType.Parameters){ // For the first item, don't add SP. if(first){ first = false; } else{ retVal.Append(" "); } retVal.Append("\"" + parameter.Name + "\" \"" + wordEncoder.Encode(parameter.Value) + "\""); } retVal.Append(")"); } else{ retVal.Append(" NIL"); } } else{ retVal.Append(" NIL"); } // contentID string contentID = entity.ContentID; if(contentID != null){ retVal.Append(" \"" + wordEncoder.Encode(contentID) + "\""); } else{ retVal.Append(" NIL"); } // contentDescription string contentDescription = entity.ContentDescription; if(contentDescription != null){ retVal.Append(" \"" + wordEncoder.Encode(contentDescription) + "\""); } else{ retVal.Append(" NIL"); } // contentEncoding if(entity.ContentTransferEncoding != null){ retVal.Append(" \"" + wordEncoder.Encode(entity.ContentTransferEncoding) + "\""); } else{ // If not specified, then must be 7bit. retVal.Append(" \"7bit\""); } // contentSize if(entity.Body is MIME_b_SinglepartBase){ retVal.Append(" " + ((MIME_b_SinglepartBase)entity.Body).EncodedData.Length.ToString()); } else{ retVal.Append(" 0"); } // envelope ---> FOR ContentType: message/rfc822 ONLY ### if(entity.Body is MIME_b_MessageRfc822){ retVal.Append(" " + IMAP_Envelope.ConstructEnvelope(((MIME_b_MessageRfc822)entity.Body).Message)); // TODO: BODYSTRUCTURE,LINES } // contentLines ---> FOR ContentType: text/xxx ONLY ### if(entity.Body is MIME_b_Text){ long lineCount = 0; StreamLineReader r = new StreamLineReader(new MemoryStream(((MIME_b_SinglepartBase)entity.Body).EncodedData)); byte[] line = r.ReadLine(); while(line != null){ lineCount++; line = r.ReadLine(); } retVal.Append(" " + lineCount.ToString()); } retVal.Append(")"); } return retVal.ToString(); }
/// <summary> /// Returns address as string value. /// </summary> /// <param name="wordEncoder">8-bit words ecnoder. Value null means that words are not encoded.</param> /// <returns>Returns address as string value.</returns> public override string ToString(MIME_Encoding_EncodedWord wordEncoder) { if(string.IsNullOrEmpty(m_DisplayName)){ return m_Address; } else{ if(wordEncoder != null && MIME_Encoding_EncodedWord.MustEncode(m_DisplayName)){ return wordEncoder.Encode(m_DisplayName) + " " + "<" + m_Address + ">"; } else{ return TextUtils.QuoteString(m_DisplayName) + " " + "<" + m_Address + ">"; } } }
/// <summary> /// Returns header field as string. /// </summary> /// <param name="wordEncoder">8-bit words ecnoder. Value null means that words are not encoded.</param> /// <param name="parmetersCharset">Charset to use to encode 8-bit characters. Value null means parameters not encoded.</param> /// <param name="reEncode">If true always specified encoding is used. If false and header field value not modified, original encoding is kept.</param> /// <returns>Returns header field as string.</returns> public override string ToString(MIME_Encoding_EncodedWord wordEncoder,Encoding parmetersCharset,bool reEncode) { if(!reEncode && m_ParseValue != null){ return m_ParseValue; } else{ if(wordEncoder != null){ return m_Name + ": " + wordEncoder.Encode(m_Value) + "\r\n"; } else{ return m_Name + ": " + m_Value + "\r\n"; } } }