本来想专心的做下WPF,最近这段时间在忙Product Demo,需要研究一下用C#代码去操作Outlook中的Meeting Request,上了一些国内外的论坛找到的东西,Demo过后就丢掉可惜了,但毕竟是花了时间的,总结分享一下,心理舒服些(很多知识来自stackoverflow,这里向大家推荐下)。这里主要总结功能的完整性和可用性。欢迎Office互操作经验多的同学补充。
闲话结束,开始正题。
在Outlook中,我们可以在Calendar里面发起一个meeting,并且可以对它修改或取消。通过Outlook的API,我们可以实现这些功能,但是很多情况下我们的服务器上并没有安装或者不允许安装Outlook,我们怎么不通过Outlook的API,用C#代码去实现这些功能?我们怎么以第三方的名义,向其他人发起一个Calendar Meeting Request?这个就是本文要讲解的情况。
流程总的说来比较简单,就是有C#构建ICS文件,然后发送到mail server,当收件人收到的时候,Outlook会解析ICS文件,做相应的操作。
1. 创建ICS格式文件。
public string BuildIcsFormatString(DateTime startTime, DateTime endTime, ICollection<string> attendees, string organizer, string subject, string description, string guid, string location) { System.Text.StringBuilder sw = new System.Text.StringBuilder(); sw.AppendLine("BEGIN:VCALENDAR"); sw.AppendLine("VERSION:2.0"); sw.AppendLine("METHOD:REQUEST"); sw.AppendLine("BEGIN:VEVENT"); if (attendees != null) { foreach (string attendee in attendees) { sw.AppendLine("ATTENDEE;ROLE=REQ-PARTICIPANT;RSVP=TRUE:mailto:" + attendee); } } sw.AppendLine("CLASS:PUBLIC"); sw.AppendLine(string.Format("CREATED:{0:yyyyMMddTHHmmssZ}", DateTime.UtcNow)); sw.AppendLine("DESCRIPTION:" + description); sw.AppendLine(string.Format("DTEND:{0:yyyyMMddTHHmmssZ}", endTime)); sw.AppendLine(string.Format("DTSTAMP:{0:yyyyMMddTHHmmssZ}", DateTime.UtcNow)); sw.AppendLine(string.Format("DTSTART:{0:yyyyMMddTHHmmssZ}", startTime)); sw.AppendLine("ORGANIZER;CN=\"NAME\":mailto:" + organizer); sw.AppendLine("SEQUENCE:0"); sw.AppendLine("UID:" + guid); sw.AppendLine("LOCATION:" + location); sw.AppendLine("SUMMARY;LANGUAGE=en-us:" + subject); sw.AppendLine("BEGIN:VALARM"); sw.AppendLine("TRIGGER:-PT720M"); sw.AppendLine("ACTION:DISPLAY"); sw.AppendLine("DESCRIPTION:Reminder"); sw.AppendLine("END:VALARM"); sw.AppendLine("END:VEVENT"); sw.AppendLine("END:VCALENDAR"); return sw.ToString(); }
要点提示1-1: ICS文件中ROLE=REQ-PARTICIPANT指的是当前的attendee类型是Required的,如果我们需要某个attendee是Optional的,要用ROLE=OPT-PARTICIPANT。所以,这段代码还可以加强用于区分Required或Optional的attendee。
要点提示1-2:我们如何通过ICS区分Create,Update?ICS是通过ID来区分不通的Calendar Meeting Request的,所以我们发出一个ICS文件,如果这个ID不在接受者的邮件系统中存在对应的Calendar Meeting Request,Outlook会执行Create操作;如果存在,Outlook会把该请求识别为一个Update操作。
要点提示1-3:我们如何通过ICS执行Delete操作?根据1-2,我们首先需要一个已经存在的ID;还有ICS文件中METHOD:REQUEST应该被修改为METHOD:CANCEL(资料上说STATUS:CANCELLED也要添加在METHOD:CANCEL之后,但是经实际测试,没有发现差别)。
2. 用SMTP发送
string meetingInfo = BuildIcsFormatString(...); System.Net.Mime.ContentType mimeType = new System.Net.Mime.ContentType("text/calendar; method=REQUEST"); AlternateView icsView = AlternateView.CreateAlternateViewFromString(meetingInfo, mimeType); MailMessage message = new MailMessage(); if (attendees != null) { foreach (string attendee in attendees) { message.To.Add(attendee); } } message.From = new MailAddress(organizer); message.AlternateViews.Add(icsView); using (SmtpClient client = new SmtpClient("MailServerName", 25)) { client.Send(message); }
3. 关于From,To,Organizer,Attendees之间的关系
其实按照步骤1和2,这个例子已经非常完整了,但是也许细心的同学会觉得From,To,Organizer和Attendees的关系是有些不清晰的,需要我们实际的测试一下。以下列出来我详细测试的结论,希望给到大家帮助。
测试前提:比如我们现在有3个人A,B,C,邮件地址分别是[email protected], [email protected], [email protected]。
3-1:MailMessage的From是必须的,如果为空会报错。但是Calendar Meeting Request的接受者并看不到From address的任何信息,接受者看到的是ICS文件中的Organizer。这就使得我们可以用第三方的名义发起一个Calendar Meeting Request。比如我可以写程序以A做为Organizer,B做为Attendee发起一个Calendar Meeting Request,B的Outlook接收以后,在B看来这个Request就是A发出的,看不到我的任何信息。
3-2:如果C在MailMessage.From,ICS中做为Organizer,A在MailMessage.To里面,但是A和B同时在ICS做为Attendees --- 只有A会收到Request。
3-3:如果C在MailMessage.From,ICS中做为Organizer,A和B在MailMessage.To里面,A和B同时在ICS做为Attendees --- A和B都会收到Request。
3-4:如果C在MailMessage.From,ICS中做为Organizer,A和B在MailMessage.To里面,只有A在ICS做为Attendees --- 只有A会收到Request。
3-5:如果C在MailMessage.From,ICS中做为Organizer,A,B,C在MailMessage.To里面,A,B,C在ICS做为Attendees --- A,B,C都会收到Request。但是在C看来,自己是Organizer,而且自己的名字不会出现在Outlook的Required或者Optional列表中。
以上结论已经足够我们正常使用,但是我相信理论上的规则会详细复杂的多,如果有同学对这方面非常了解,欢迎补充和指正。
4. 补充疑问
还没有解决的问题:Organizer可以收到Request,但是Organizer的Calendar是没有办法被相应Update的。 Outlook发现当前接收者就是Organizer的时候,是不会让accept这个Request的,导致第三方程序用其他人的名义发起一个Request的时候,发起人的Calendar不能被相应更新。这个问题我已经发在MSDN上和stackoverflow上,都没有回应,希望有高手帮助解答。