Skip to the content.

Teams Bot如何解析和发送 at 用户

之前有好些看博客的开发者问我有什么较好的方法来解析 at 信息。用户在channel里发消息给 bot 的时候,通常需要 at bot的名字,当然在消息文字中可能还会 at 其他用户,比如 bot的名字是 MyBot,那 bot 收到的 Activity 里的 Text 值类似于:

<at>MyBot</at> 请帮我查看一下 <at>Tom</at> 的明天有空时间。

可以看到 Teams 使用了 at tag 来把用户 at 的信息包含起来。

看到这里,大家可能会想,这个简单啊,可以分析一下字符串,把 <at> 的内容find到,然后在找到结束的 tag </at> 。但是如果你仔细想想,怎么写循环,也不简单把,至少需要 5 到 10 行的c#代码,可能大家还会想到使用正则表达式,对,正则会简单一些,但是你要快速的写出这个正则表达式也不是特别简单的事情。

另外,我们还需要考虑一些转义的字符,是不是瞬间觉得也不简单了。

解析 at 信息

好在 bot 的 sdk 给我们提供了一个简单的方法。

protected override async Task OnMessageActivityAsync(ITurnContext<IMessageActivity> turnContext, CancellationToken cancellationToken)
{
    Mention[] mentions = turnContext.Activity.GetMentions();
    if(mentions != null)
    {
        foreach (var mention in mentions)
        {
            ...
        }
    }
}

GetMentions 是一个 Activity 的扩展方法,里面具体的实现可以到 sdk 的源代码里找到。这个方法会返回一个 Mention 的数组

public class Mention
{
    ...
    public ChannelAccount Mentioned { get; set; }
    ...
}

public class ChannelAccount
{
    ...
    public string Name { get; set; }
    ...
}

从上面的代码可以看到,我可以通过 Mentioned 的 Name 来获取被 at 的对象的名字。

var name = mention.Mentioned.Name

发送 at 信息

除了解析,sdk 当然也提供了发送 at 信息的方法,如下:

protected override async Task OnMessageActivityAsync(ITurnContext<IMessageActivity> turnContext, CancellationToken cancellationToken)
{
    var mention = new Mention
    {
        Mentioned = turnContext.Activity.From,
        Text = $"<at>{XmlConvert.EncodeName(turnContext.Activity.From.Name)}</at>",
    };

    var replyActivity = MessageFactory.Text($"Hello {mention.Text}.");
    replyActivity.Entities = new List<Entity> { mention };

    await turnContext.SendActivityAsync(replyActivity, cancellationToken);
}

从上面的额代码里可以看到,发送 at 信息并不是很简单,还是需要我们自己来组建一个 Mention 的对象,然后把这个对象放如到回复的 activity 的 Entities 属性里。

特别需要注意的一点是:需要 at 的人的名字,需要做 xml encode,来确保不会因为 name 里有特殊的字符,而导致整个text不合法。

看到这里,大家是不是对 at 的机制有了进一步的深入理解? :)

Written on October 14, 2021