今天实验了一下使用EWS访问exchange邮件,网上说的都很简单,可我却是碰到了N多问题啊,不过最终还是成功了,也不枉我加班了。把一些方法及遇到的问题记录下来,希望对有类似exchange功能开发需求的人有所帮助。
第一步是下载Exchange Web Services Managed API,这个DLL封装了很多对EWS的访问,比起直接使用EWS生成的代理类,那是方便多了,下载地址:
http://www.microsoft.com/downloads/en/details.aspx?FamilyID=c3342fb3-fbcc-4127-becf-872c746840e1
使用EWS的代码示例网上有不少,微软站点上也有,博客园里有两篇关于这个的介绍(下篇未给出链接):
http://www.cnblogs.com/diaojia/archive/2010/10/19/1855839.html
代码还是比较简单的,在此我贴上我的测试代码:
public class AccessInfo { public string UserName; public string Password; public string Domain; public string ServerUrl; public string Email; } class Program { static void Main(string[] args) { AccessInfo info = new AccessInfo() { UserName = "administrator", Password = "p@ssw0rd", Domain = "contoso.com", ServerUrl = "https://contoso-exchange.contoso.com/ews/Exchange.asmx Email = "[email protected]" }; ReadMail(info); Console.Read(); } static void ReadMail(AccessInfo Info) { //ExchangeService版本为2010 ExchangeService service = new ExchangeService(ExchangeVersion.Exchange2010); //参数是用户名,密码,域 service.Credentials = new WebCredentials(Info.UserName, Info.Password, Info.Domain); //给出Exchange Server的URL http://xxxxxxx service.Url = new Uri(Info.ServerUrl); //你自己的邮件地址 [email protected] //service.AutodiscoverUrl(Info.Email); //创建过滤器, 条件为邮件未读. SearchFilter sf = new SearchFilter.IsEqualTo(EmailMessageSchema.IsRead, false); //查找Inbox,加入过滤器条件,结果10条 FindItemsResults<Item> findResults = null; try { findResults = service.FindItems(WellKnownFolderName.Inbox, sf, new ItemView(10)); } catch (Exception ex) { Console.WriteLine(ex.Message); } foreach (Item item in findResults.Items) { EmailMessage email = EmailMessage.Bind(service, item.Id); Console.WriteLine(email.Subject); } } }",
这段代码是最终测试成功的代码,下面列一下碰到过的问题。
1、在service.AutodiscoverUrl(Info.Email)处出现异常:AutodiscoverUrl could not be located
这个问题最终没有得到根本解决,我只是简单的把这一行注释了,因为我想既然显示的指定了service.Url,那也就没必要autodiscover了;
2、在service.FindItems(WellKnownFolderName.Inbox, sf, new ItemView(10))处出现异常,提示Http404错误
这个问题郁闷了很久,因为写的URL明明是没有错的,在浏览器中都可以访问,也可以使用它来生成代理类,后面将要放弃的时候,找到了跟我有同样遭遇的人写
的帖子:http://www.cnblogs.com/Bear-Study-Hard/archive/2010/10/11/1847824.html。(又是博客园),里头说到是因为EWS站点禁用了SSL的缘故。的确
我的EWS站点也是禁用了SSL的,我比较不喜欢在浏览器里访问时老提示证书无效的警告。于是乎把SSL加上了,同时把http改为https(一开始测试是写的http)。
3、启用了SSL,在FindItems的时候提示“未能为SSL/TLS安全通道建立信任关系”。
完了,证书的问题来了,我一直都不明白的东西,始终不清楚怎样为IIS站点设置证书,以及客户端安装证书。google到了两篇文章,一篇是win2003 IIS6的,
一篇是win2008 IIS7的:
win2003 IIS6:http://www.cnblogs.com/kimzeng/archive/2009/12/17/1626171.html
win2008 IIS7:http://blog.csdn.net/steel_ligang/archive/2009/09/25/4593102.aspx
在此我要鄙视一下MS,两代产品之间老是搞颠覆性的操作方式。两篇文章里介绍了很多,但都同时缺少了关于怎样提交所申请的证书,以及如何获取到颁发的证书的
操作。下面简单介绍一下:
提交申请的证书
在浏览器中打开:http://contoso.com/certsrv/,其中contoso.com是CA服务器机器名,也就是域控机器名,界面类似如下图所示:
点击“申请一个证书”,然后选择“高级证书申请”,然后再选择“使用 base64 编码的 CMC 或 PKCS #10 文件提交 一个证书申请,或使用 base64 编码的 PKCS #7 文件续订证书申请。”,最中打开的页面如下图所示:
在上面的文本框里贴上生成的HASH码文本文件里的所有内容,然后点击提交,证书申请动作才算完成。
之后打开证书颁发机构,找到“挂起的申请”,然后在右边的列表里找到申请的证书,右击——所有任务——颁发,就完成了。
新申请的证书可到证书管理首页(http://contoso.com/certsrv/)里选择“查看挂起的证书申请的状态”,在接下来的页面里有下载链接提供。
在此我要提醒一下,必须在安装证书服务之前安装IIS,否则就没有WEB上的证书服务页面。
回到第3点提到的问题,在折腾了很长时间之后,在我的IIS里依然没有成功的加上证书。后面找到了微软的一篇文章:
http://support.microsoft.com/kb/981954/zh-cn
里面提到了关键的一点:过程中创建一个自签名或 $ self-issued 的根证书。尽管这一句并不是出现在这篇文章里所描述的解决办法里,然而却提醒了我,可以为IIS
创建一个自签名的证书,或许有用。IIS7中服务器证书管理界面中就有创建自签名证书的功能,过程就不细说了。创建完成之后,记得更改EWS站点所绑定的证书为刚才所创建的证书。更改之后,请确保程序中指定的服务URL域名与证书中的“颁发给”属性一致。
最后一点要提醒的,不要随便为EWS站点添加主机头,那将可能会导致EWS无法启动,除非你对WCF的配置文件很熟悉。