MS Exchange Server(社内用)の特定のメールボックスから電子メールを監視して読み取る機能が必要です。また、送信者の電子メールアドレス、件名、メッセージ本文を読み、添付ファイルがあればダウンロードできるようにする必要もあります。
C#(またはVB.NET)を使用してこれを行う最良の方法は何ですか?
MS Exchange Server(社内用)の特定のメールボックスから電子メールを監視して読み取る機能が必要です。また、送信者の電子メールアドレス、件名、メッセージ本文を読み、添付ファイルがあればダウンロードできるようにする必要もあります。
C#(またはVB.NET)を使用してこれを行う最良の方法は何ですか?
回答:
それは混乱です。.NET相互運用DLLを介したMAPIまたはCDOは、Microsoftによって正式にサポートされていません-正常に動作するように見えますが、メモリモデルが異なるため、メモリリークに問題があります。CDOEXを使用することもできますが、これはリモートではなく、Exchangeサーバー自体でのみ機能します。役に立たない。Outlookと相互運用できますが、これでOutlookに依存するようになりました。やりすぎ。最後に、Exchange 2003のWebDAVサポートを使用することもできますが、WebDAVは複雑で、.NETの組み込みサポートは不十分であり、(侮辱を加えるために)Exchange 2007 は WebDAVサポートをほぼ完全に削除します。
やるべきことは何ですか?結局、AfterLogicのIMAPコンポーネントを使用して、IMAP経由でExchange 2003サーバーと通信しましたが、これは非常にうまく機能していました。(私は通常、無料またはオープンソースのライブラリを探していますが、特に2003年のIMAP実装のいくつかの奇妙な点に関しては、すべての.NETライブラリが必要であることがわかりましたが、これは十分に安価で、最初の試してみてください。他にもあります。)
ただし、組織がExchange 2007を使用している場合は、幸運です。Exchange 2007にはSOAPベースのWebサービスインターフェイスが付属しており、最終的にはExchangeサーバーと対話する、言語に依存しない統合された方法を提供します。2007+を要件とすることができれば、これは間違いなく進むべき道です。(残念ながら、私の会社には「しかし2003年は破られない」という方針があります。)
Exchange 2003と2007の両方をブリッジする必要がある場合は、IMAPまたはPOP3が間違いなく適しています。
ええと、
私はここでは少し遅すぎるかもしれませんが、これはEWSにとってちょっとポイントではありませんか?
https://msdn.microsoft.com/en-us/library/dd633710(EXCHG.80).aspx
メールボックスからメールを取得するには、約6行のコードが必要です。
ExchangeService service = new ExchangeService(ExchangeVersion.Exchange2007_SP1);
//service.Credentials = new NetworkCredential( "{Active Directory ID}", "{Password}", "{Domain Name}" );
service.AutodiscoverUrl( "First.Last@MyCompany.com" );
FindItemsResults<Item> findResults = service.FindItems(
WellKnownFolderName.Inbox,
new ItemView( 10 )
);
foreach ( Item item in findResults.Items )
{
Console.WriteLine( item.Subject );
}
service.autodiscoverurl
、私はを入力する必要がありservice.credentials
ますか?
現在推奨されている(Exchange 2013および2016)APIはEWSです。純粋にHTTPベースであり、どの言語からでもアクセスできますが、.Netと Java固有のライブラリます。
拡張MAPI。これは、Outlookで使用されるネイティブAPIです。MSEMS
最終的には、Exchange MAPIプロバイダーを使用します。これは、RPC(Exchange 2013ではサポートされなくなった)、RPC-over-HTTP(Exchange 2007以降)またはMAPI-over-HTTP(Exchange 2013以降)を使用してExchangeと通信できます。
API自体には、アンマネージC ++またはDelphiからのみアクセスできます。Redemption(任意の言語)を使用することもできます。そのRDOオブジェクトファミリーはExtended MAPIラッパーです。拡張MAPIを使用するには、Outlookまたはスタンドアロン(Exchange)バージョンのMAPIをインストールする必要があります(拡張サポート上にあり、Unicode PSTおよびMSGファイルをサポートせず、Exchange 2016にアクセスできません)。サービスで拡張MAPIを使用できます。
OutlookSpyまたはMFCMAPIを使用してAPIで遊ぶことができます。
Outlookオブジェクトモデル -Exchange固有ではありませんが、コードが実行されるマシン上のOutlookで利用可能なすべてのデータにアクセスできます。サービスでは使用できません。
Exchange Active Sync。マイクロソフトは、このプロトコルに重要なリソースを投資しなくなりました。
OutlookはCDO 1.21ライブラリ(拡張MAPIをラップする)をインストールするために使用されていましたが、Microsoftによって廃止されており、更新を受け取りません。
以前はMAPI33と呼ばれるサードパーティの.Net MAPIラッパーがありましたが、現在は開発もサポートもされていません。
WebDAV-非推奨。
Exchangeのコラボレーティブデータオブジェクト(CDOEX)-非推奨。
Exchange OLE DBプロバイダー(EXOLEDB)-非推奨。
これは、WebDAVを実行するために置いていた古いコードです。Exchange 2003に対して書かれたものだと思いますが、もう覚えていません。参考になればお気軽にご利用ください...
class MailUtil
{
private CredentialCache creds = new CredentialCache();
public MailUtil()
{
// set up webdav connection to exchange
this.creds = new CredentialCache();
this.creds.Add(new Uri("http://mail.domain.com/Exchange/me@domain.com/Inbox/"), "Basic", new NetworkCredential("myUserName", "myPassword", "WINDOWSDOMAIN"));
}
/// <summary>
/// Gets all unread emails in a user's Inbox
/// </summary>
/// <returns>A list of unread mail messages</returns>
public List<model.Mail> GetUnreadMail()
{
List<model.Mail> unreadMail = new List<model.Mail>();
string reqStr =
@"<?xml version=""1.0""?>
<g:searchrequest xmlns:g=""DAV:"">
<g:sql>
SELECT
""urn:schemas:mailheader:from"", ""urn:schemas:httpmail:textdescription""
FROM
""http://mail.domain.com/Exchange/me@domain.com/Inbox/""
WHERE
""urn:schemas:httpmail:read"" = FALSE
AND ""urn:schemas:httpmail:subject"" = 'tbintg'
AND ""DAV:contentclass"" = 'urn:content-classes:message'
</g:sql>
</g:searchrequest>";
byte[] reqBytes = Encoding.UTF8.GetBytes(reqStr);
// set up web request
HttpWebRequest request = (HttpWebRequest)HttpWebRequest.Create("http://mail.domain.com/Exchange/me@domain.com/Inbox/");
request.Credentials = this.creds;
request.Method = "SEARCH";
request.ContentLength = reqBytes.Length;
request.ContentType = "text/xml";
request.Timeout = 300000;
using (Stream requestStream = request.GetRequestStream())
{
try
{
requestStream.Write(reqBytes, 0, reqBytes.Length);
}
catch
{
}
finally
{
requestStream.Close();
}
}
HttpWebResponse response = (HttpWebResponse)request.GetResponse();
using (Stream responseStream = response.GetResponseStream())
{
try
{
XmlDocument document = new XmlDocument();
document.Load(responseStream);
// set up namespaces
XmlNamespaceManager nsmgr = new XmlNamespaceManager(document.NameTable);
nsmgr.AddNamespace("a", "DAV:");
nsmgr.AddNamespace("b", "urn:uuid:c2f41010-65b3-11d1-a29f-00aa00c14882/");
nsmgr.AddNamespace("c", "xml:");
nsmgr.AddNamespace("d", "urn:schemas:mailheader:");
nsmgr.AddNamespace("e", "urn:schemas:httpmail:");
// Load each response (each mail item) into an object
XmlNodeList responseNodes = document.GetElementsByTagName("a:response");
foreach (XmlNode responseNode in responseNodes)
{
// get the <propstat> node that contains valid HTTP responses
XmlNode uriNode = responseNode.SelectSingleNode("child::a:href", nsmgr);
XmlNode propstatNode = responseNode.SelectSingleNode("descendant::a:propstat[a:status='HTTP/1.1 200 OK']", nsmgr);
if (propstatNode != null)
{
// read properties of this response, and load into a data object
XmlNode fromNode = propstatNode.SelectSingleNode("descendant::d:from", nsmgr);
XmlNode descNode = propstatNode.SelectSingleNode("descendant::e:textdescription", nsmgr);
// make new data object
model.Mail mail = new model.Mail();
if (uriNode != null)
mail.Uri = uriNode.InnerText;
if (fromNode != null)
mail.From = fromNode.InnerText;
if (descNode != null)
mail.Body = descNode.InnerText;
unreadMail.Add(mail);
}
}
}
catch (Exception e)
{
string msg = e.Message;
}
finally
{
responseStream.Close();
}
}
return unreadMail;
}
}
そしてmodel.Mail:
class Mail
{
private string uri;
private string from;
private string body;
public string Uri
{
get { return this.uri; }
set { this.uri = value; }
}
public string From
{
get { return this.from; }
set { this.from = value; }
}
public string Body
{
get { return this.body; }
set { this.body = value; }
}
}
CodeProject.comで公開されているコードを使用しました。POP3を使用する場合は、私が見つけた優れたソリューションの1つです。
MAPIを使用してメールボックスにアクセスし、必要な情報を取得できる必要があります。残念ながら、私が知っている唯一の.NET MAPIライブラリ(MAPI33)はメンテナンスされていないようです。これは、以前は.NETを介してMAPIにアクセスするための優れた方法でしたが、今はその有効性について話すことはできません。ここで入手できる場所に関する詳細情報があります。MAPI33.dllのダウンロード場所?
私は最終的にRedemptionを使用して機能する解決策を得ました、これらの質問を見てください...
1つのオプションはOutlookを使用することです。Exchangeサーバーにアクセスし、インターフェイスとしてOutlookを使用するメールマネージャーアプリケーションがあります。その汚いがそれは動作します。
コード例:
public Outlook.MAPIFolder getInbox()
{
mailSession = new Outlook.Application();
mailNamespace = mailSession.GetNamespace("MAPI");
mailNamespace.Logon(mail_username, mail_password, false, true);
return MailNamespace.GetDefaultFolder(Outlook.OlDefaultFolders.olFolderInbox);
}