You need to enable JavaScript to run this app.
最新活动
大模型
产品
解决方案
定价
生态与合作
支持与服务
开发者
了解我们

如何在C#中发送ICS文件并通过回调获取用户选择结果?

嘿,我来帮你解决这个问题!要在C#里发送ICS日历邀请并获取用户的选择结果,核心是利用iCalendar的RSVP机制搭配一个回调接口来接收用户的响应。下面一步步给你讲明白怎么做:

1. 生成带RSVP要求的ICS文件

首先得生成一个能触发用户回复的ICS文件,关键是开启RSVP功能,让邮件客户端显示接受/拒绝/暂定的选项。手动写ICS容易出错,推荐用NuGet包iCalendar.NET来生成:

using Ical.Net;
using Ical.Net.CalendarComponents;
using Ical.Net.DataTypes;
using Ical.Net.Serialization;

public string GenerateRsvpEnabledInvite(string attendeeEmail, string meetingUniqueId)
{
    var calendar = new Calendar();
    var meetingEvent = new CalendarEvent
    {
        Start = new CalDateTime(DateTime.Now.AddDays(1), "UTC"),
        End = new CalDateTime(DateTime.Now.AddDays(1).AddHours(1), "UTC"),
        Summary = "项目同步会议",
        Description = "讨论Q3项目进度与风险",
        Location = "线上会议室(链接:xxx)",
        Organizer = new Organizer("mailto:team@yourcompany.com"),
        RequiresReply = true, // 强制要求用户回复
        Uid = meetingUniqueId // 设置唯一会议ID,后续回调时用来验证
    };

    // 添加参会者并开启RSVP
    var attendee = new Attendee($"mailto:{attendeeEmail}")
    {
        Rsvp = true,
        ParticipationStatus = "NEEDS-ACTION" // 初始状态:待回复
    };
    meetingEvent.Attendees.Add(attendee);

    // 标记这是一个邀请请求
    calendar.Method = "REQUEST";
    calendar.Events.Add(meetingEvent);

    // 序列化为ICS字符串
    var serializer = new CalendarSerializer();
    return serializer.SerializeToString(calendar);
}
2. 发送带ICS附件的邮件

推荐用MailKitMimeKit发送邮件(SmtpClient已过时),记得设置正确的MIME类型,让邮件客户端识别为日历邀请:

using MailKit.Net.Smtp;
using MimeKit;
using System.Text;

public async Task SendCalendarInvite(string toEmail, string icsContent)
{
    var message = new MimeMessage();
    message.From.Add(new MailboxAddress("项目组", "team@yourcompany.com"));
    message.To.Add(new MailboxAddress(toEmail.Split('@')[0], toEmail));
    message.Subject = "【邀请】项目同步会议";

    var bodyBuilder = new BodyBuilder();
    bodyBuilder.TextBody = "请查看附件的会议邀请,点击邮件内的按钮回复你的参会状态。";

    // 添加ICS附件,设置关键头信息
    var icsAttachment = new MimePart("text", "calendar")
    {
        Content = new MimeContent(new MemoryStream(Encoding.UTF8.GetBytes(icsContent))),
        ContentDisposition = new ContentDisposition(ContentDisposition.Attachment),
        ContentTransferEncoding = ContentEncoding.Base64,
        FileName = "meeting-invite.ics"
    };
    // 告诉客户端这是邀请请求
    icsAttachment.Headers.Add("Method", "REQUEST");
    icsAttachment.Headers.Add("Content-Class", "urn:content-classes:calendarmessage");

    bodyBuilder.Attachments.Add(icsAttachment);
    message.Body = bodyBuilder.ToMessageBody();

    // 发送邮件(替换为你的SMTP配置)
    using var client = new SmtpClient();
    await client.ConnectAsync("smtp.yourcompany.com", 587, MailKit.Security.SecureSocketOptions.StartTls);
    await client.AuthenticateAsync("smtp-username", "smtp-password");
    await client.SendAsync(message);
    await client.DisconnectAsync(true);
}
3. 接收用户响应的回调接口

当用户点击接受/拒绝/暂定后,邮件客户端会自动发送一个POST请求到你的公网可访问接口,请求体是包含用户选择的ICS内容。你需要解析这个ICS来获取用户的选择:

以ASP.NET Core为例,创建一个回调控制器:

using Microsoft.AspNetCore.Mvc;
using Ical.Net;
using Ical.Net.CalendarComponents;

[ApiController]
[Route("api/calendar")]
public class CalendarRsvpController : ControllerBase
{
    [HttpPost("rsvp")]
    public async Task<IActionResult> ReceiveRsvpResponse()
    {
        // 读取请求体的ICS内容
        using var reader = new StreamReader(Request.Body);
        var icsContent = await reader.ReadToEndAsync();

        // 解析ICS
        var calendar = Calendar.Load(icsContent);
        var meetingEvent = calendar.Events.FirstOrDefault();
        if (meetingEvent == null)
        {
            return BadRequest("无效的会议响应内容");
        }

        // 获取参会者和选择状态
        var attendee = meetingEvent.Attendees.FirstOrDefault();
        if (attendee == null)
        {
            return BadRequest("未找到参会者信息");
        }

        // PARTSTAT的核心值:ACCEPTED(接受)、DECLINED(拒绝)、TENTATIVE(暂定)
        string userChoice = attendee.ParticipationStatus;
        string attendeeEmail = attendee.Value.Replace("mailto:", "");
        string meetingId = meetingEvent.Uid;

        // 这里可以把结果存入数据库、触发通知等业务逻辑
        Console.WriteLine($"会议[{meetingId}]:参会者{attendeeEmail}选择了{userChoice}");

        // 返回成功响应,告知客户端已接收
        return Ok();
    }
}
4. 重要注意事项
  • 接口可访问性:回调接口必须是公网可访问的HTTPS地址(多数邮件客户端拒绝HTTP请求)。
  • 请求验证:利用ICS里的Uid(唯一会议ID)验证请求合法性,防止伪造响应。
  • 兼容性:部分小众邮件客户端可能不支持自动回调,建议在邮件正文里补充手动回复的方式。
  • 时区处理:生成ICS时务必指定时区(比如示例中的UTC),避免会议时间显示错误。

内容的提问来源于stack exchange,提问作者Daniel Fernandes

火山引擎 最新活动