Отправка / отправка электронных писем с дублирующейся строкой темы с использованием WebDav в C # - PullRequest
1 голос
/ 08 марта 2011

У меня есть некоторый код, который использует метод WEBDAV 'SEARCH' для извлечения электронной почты из папки 'Inbox' почтовых ящиков Exchange - мой код принимает innerXML WEBresponse HTTPWebRquests.эти namsspaces:

'urn: схемы: httpmail' & urn: схемы: mailheader

Позволяет мне извлечь элементы:

f: текстовое описание d: fromd: subject f: datareceived ... и т. д.

Затем я создаю коллекцию этих деталей в списке и, используя 'Subject' вместе с URI, использую метод 'PUT', чтобы воссоздать эти сообщения в папке 'draft'перед использованием «MOVE» для отправки писем (помещает их в отправленные элементы с помощью оператора «/ ## DavMailSubmissionURI ## /" ».

Проблема, с которой я сталкиваюсь, заключается в характере писем, с которыми я имею делоимеют тенденцию приходить с одной и той же темой, поэтому путают с тем, какие письма были отправлены / нет.

Кто-нибудь знает способ обойти это, я не знаю, почему «PUT» полагается наСтрока темы для URI почтовому ресурсу, а не тэг HREF, который является уникальным.Любые идеи:

Код ниже:

public class EmailReaderWebDav
{

    public enum enmHTTPType 
    {
        HTTP,
        HTTPS,
    }

    private String strServer { get; set; }      //"mail1"              ------ Exchange server name
    public String strPassword { get; set; }     //"password"           ------ Account Domain Password
    public String strDomain { get; set; }       //"mydocmian"          ------ Domain
    public String strMailBox { get; set; }      //"mymailbox"          ------ UserName
    public String mailFolder { get; set; }      //"inbox"              ------ Mail Folder 
    private String httpProtocol { get; set; }   //http:// ? or https://
    private String mailboxURI { get; set; }     //httpprotocol// + strserver + "/exhange/" + strmailbox

    public List<MailStruct > ListOfEmailDetails { get; private set; }

    private String strQuerySearch { get; set; }

    public EmailReaderWebDav(String serverName, String domain, String mailBox, String password, String mailmailFolder,enmHTTPType HTTPType)
    {
        strServer = serverName;
        strDomain = domain;
        strMailBox = mailBox;
        strPassword = password;
        mailFolder = mailmailFolder;

        httpProtocol = (HTTPType == enmHTTPType.HTTPS) ? "https://" : "http://";
        mailboxURI = httpProtocol + strServer + "/exchange/"  + strMailBox + "/inbox/";

    }


    public void forwardEmails(List<MailStruct> emailsToSend)
    {

        emailsToSend.ForEach(x => SendEmail(x,enmHTTPType.HTTP ));

    }

    public void MakeListofEmailsToForward()
    {

        String tmpQuery =                   

            "<?xml version=\"1.0\"?>"
                + "<D:searchrequest xmlns:D = \"DAV:\" >"
                + "<D:sql>"

                    + " SELECT "
                    + "\"urn:schemas:mailheader:to\","
                    + "\"urn:schemas:mailheader:from\","
                    + "\"urn:schemas:mailheader:subject\","
                    + "\"urn:schemas:httpmail:datereceived\","
                    + "\"urn:schemas:httpmail:textdescription\""
                    + " FROM \"" + mailboxURI + "\""
                    + " WHERE \"DAV:ishidden\" = false AND \"DAV:isfolder\" = false"
                    + "</D:sql>"
            + "</D:searchrequest>";


            // Search Request to get emails from target folder.
        HttpWebRequest SearchRequest = MakeWebRequest("SEARCH", "text/xml", mailboxURI);

        Byte[] bytes = Encoding.UTF8.GetBytes((String)tmpQuery);
            SearchRequest.ContentLength = bytes.Length;

        Stream SearchRequestStream = SearchRequest.GetRequestStream();

            SearchRequestStream.Write(bytes, 0, bytes.Length);
            SearchRequestStream.Close();

                // get the webresponse from the searchrequest.
        WebResponse SearchResponse = MakeWebResponse(SearchRequest);

        String EmailsInXML = extractXMLFromWebResponse(SearchResponse);

            ListOfEmailDetails = extractMailPropertiesFromXMLString(EmailsInXML);

    }

    public void SendEmail(MailStruct mailToForward, enmHTTPType HTTPType)
    {

        String submissionUri = httpProtocol + strServer + "/" + "exchange"  + "/" + strMailBox + "/##DavMailSubmissionURI##/";
        String draftsUri = httpProtocol + strServer + "/" +"exchange" + "/" + strMailBox + "/Drafts/" + mailToForward.Subject + ".eml";

        String message = "To: " + mailToForward.To + "\n"
            + "Subject: " + mailToForward.Subject + "\n"
            + "Date: " + mailToForward.Received 
            + "X-Mailer: mailer" + "\n"
            + "MIME-Version: 1.0" + "\n"
            + "Content-Type: text/plain;" + "\n"
            + "Charset = \"iso-8859-1\"" + "\n"
            + "Content-Transfer-Encoding: 7bit" + "\n"
            + "\n" + mailToForward.MailBody;

        // Request to put an email the drafts folder.
        HttpWebRequest putRequest = MakeWebRequest("PUT", "message/rfc822",draftsUri );

        Byte[] bytes = Encoding.UTF8.GetBytes((String)message);
        putRequest.Headers.Add("Translate", "f");
        putRequest.Timeout = 300000;
        putRequest.ContentLength = bytes.Length;

        Stream putRequestStream = putRequest.GetRequestStream();

        putRequestStream.Write(bytes, 0, bytes.Length);
        putRequestStream.Close();



        // Put the message in the Drafts folder of the sender's mailbox.
        HttpWebResponse putResponse = MakeWebResponse(putRequest);

        putResponse.Close();

        // Request to move the email from the drafts to the mail submission Uri.
        HttpWebRequest moveRequest = MakeWebRequest("MOVE", "text/xml", draftsUri);

        moveRequest.Headers.Add("Destination", submissionUri);


        // Put the message in the mail submission folder.
        HttpWebResponse moveResponse = MakeWebResponse(moveRequest);
            moveResponse.Close();

    }


    private CredentialCache getCredentials(String URI)
    {
        CredentialCache tmpCreds = new CredentialCache();

            tmpCreds.Add(new Uri(URI), "NTLM", new NetworkCredential(strMailBox, strPassword,strDomain ));

            ServicePointManager.ServerCertificateValidationCallback = delegate(object sender, System.Security.Cryptography.X509Certificates.X509Certificate pCertificate, System.Security.Cryptography.X509Certificates.X509Chain pChain, System.Net.Security.SslPolicyErrors pSSLPolicyErrors)
            {
                return true;
            };

        return tmpCreds;
    }

    private HttpWebRequest MakeWebRequest(String method,String contentType,String URI)
    {

        HttpWebRequest tmpWebRequest;
        tmpWebRequest  = (HttpWebRequest)HttpWebRequest.Create(URI);
        tmpWebRequest.Credentials = getCredentials (URI);
        tmpWebRequest.Method = method;
        tmpWebRequest.ContentType = contentType ;

            return tmpWebRequest ;
    }

    private HttpWebResponse MakeWebResponse(HttpWebRequest webRequest)
    {
        HttpWebResponse tmpWebresponse = (HttpWebResponse)webRequest.GetResponse();

            return tmpWebresponse;
    }


    WebResponse getMailsFromWebRequest(String strRootURI, String strQuerySearch)
    {

        HttpWebRequest SEARCHRequest;
        WebResponse SEARCHResponse;
        CredentialCache MyCredentialCache;
        Byte[] bytes = null;
        Stream SEARCHRequestStream = null;

        try
        {

            MyCredentialCache = new CredentialCache();
            MyCredentialCache.Add(new Uri(strRootURI ), "NTLM", new NetworkCredential(strMailBox.ToLower(), strPassword, strDomain));

            SEARCHRequest = (HttpWebRequest)HttpWebRequest.Create(strRootURI );

            ServicePointManager.ServerCertificateValidationCallback = delegate(object sender, System.Security.Cryptography.X509Certificates.X509Certificate pCertificate, System.Security.Cryptography.X509Certificates.X509Chain pChain, System.Net.Security.SslPolicyErrors pSSLPolicyErrors)
            {
                return true;
            };

            SEARCHRequest.Credentials = MyCredentialCache;
            SEARCHRequest.Method = "SEARCH";
            SEARCHRequest.ContentType = "text/xml";

            bytes = Encoding.UTF8.GetBytes((string)strQuerySearch);

           SEARCHRequest.ContentLength = bytes.Length;
           SEARCHRequestStream = SEARCHRequest.GetRequestStream();


           SEARCHRequestStream.Write(bytes, 0, bytes.Length);
           SEARCHResponse =(HttpWebResponse ) SEARCHRequest.GetResponse();
           SEARCHRequestStream.Close();
           SEARCHRequest.Timeout = 300000;

            System.Text.Encoding enc = System.Text.Encoding.Default;

                if (SEARCHResponse == null)
                {
                    Console.WriteLine("Response returned NULL!");
                }
                else
                {
                    Console.WriteLine(SEARCHResponse.ContentLength);

                }

                    return SEARCHResponse;

        }

        catch (Exception ex)
        {
            Console.WriteLine("Problem: {0}", ex.Message);
                return null;

        }
    }


    private String extractXMLFromWebResponse(WebResponse SearchResponse)
    {
        String tmpStream;

        using(StreamReader strmReader = new StreamReader(SearchResponse.GetResponseStream(), System.Text.Encoding.ASCII))
        {
            tmpStream  = strmReader.ReadToEnd();

            strmReader.Close();
        }

            return tmpStream;

    }

    private List<MailStruct > extractMailPropertiesFromXMLString(String strXmlStream)
    {

        List<MailStruct> tmpListOfMailProperties = new List<MailStruct>();
        XmlDocument doc = new XmlDocument();
        doc.InnerXml = strXmlStream ;

        XmlNamespaceManager xmlNameSpaces = new XmlNamespaceManager(doc.NameTable);

        xmlNameSpaces.AddNamespace("a", "DAV:");
        xmlNameSpaces.AddNamespace("f", "urn:schemas:httpmail:");
        xmlNameSpaces.AddNamespace("d", "urn:schemas:mailheader:");

        XmlNodeList mailNodes = doc.SelectNodes("//a:propstat[a:status='HTTP/1.1 200 OK']/a:prop", xmlNameSpaces);

        foreach (XmlElement node in mailNodes)
        {

            tmpListOfMailProperties.Add(new MailStruct()
                            {
                                MailBody = node.SelectSingleNode("//f:textdescription",xmlNameSpaces ).InnerText ,
                                from = node.SelectSingleNode ("//d:from",xmlNameSpaces ).InnerText ,
                                To = "dfoster@liquidcapital.com",
                                Subject = node.SelectSingleNode("//d:subject",xmlNameSpaces ).InnerText.ToString () ,
                                Received = node.SelectSingleNode ("//f:datereceived",xmlNameSpaces ).InnerText.ToString ()
                            }
                         );
        }

        return tmpListOfMailProperties;

    }


    public struct MailStruct
    {
        public String To { get; set; }
        public String from { get; set; }
        public String Subject { get; set; }
        public String Received { get; set; }
        public String MailBody { get; set; }
    }
}

}

Ответы [ 2 ]

0 голосов
/ 08 марта 2011

Похоже, вы работаете с Exchange 2003. Имейте в виду, что WebDAV больше не поддерживается в Exchange версии 2010 ... и с 2007 и новее вы можете использовать WSDL, чтобы делать все, что вам нужно.Все, что вам нужно, это Exchange 2007 CAS, чтобы сделать это.

То, что я упомянул, является более долгосрочным подходом и требует меньше головной боли, чем разбор неподдерживаемого WEBDAV XML

.свойство, уникальное для каждого сообщения.Используйте MFCMapi (на codeplex), чтобы найти его.Свойство будет называться «MessageURL» или что-то в этом роде.Используйте это свойство в URL для ваших вызовов webdav.

0 голосов
/ 08 марта 2011

Использование только темы для идентификации электронной почты, действительно, не очень хороший способ. Если я помню, что это правильно, не существует автоматического / очевидного идентификатора электронной почты для exchange / webdav?

Если вы проанализируете тему, чтобы выбрать сообщение, я бы также выбрал дополнительную информацию о почтовом конверте - размер, длину, чтобы создать свой собственный идентификатор. Лучший шаг должен состоять в том, чтобы вы создали какой-то хэш (проверьте криптографию) из всего тела сообщения ИЛИ, т.е. первого символа первого хх слова в теле письма (хотя и более тяжелая обработка). Это будет приводить к тому же хеш-значению каждый раз, когда вы вызываете его в одном и том же почтовом конверте. До тех пор, пока содержимое письма не будет оставлено без изменений.

...