调用 GetConversationMembersAsync 进行 1:1 私人对话时,操作返回无效状态代码“禁止”

2024-01-30

使用此处的“Hello World”Microsoft Teams 应用程序示例:https://github.com/OfficeDev/msteams-samples-hello-world-csharp https://github.com/OfficeDev/msteams-samples-hello-world-csharp

在调用来自消息传递扩展的操作命令后,尝试获取个人 Microsoft Teams 1:1 聊天的参与者列表。具体来说,我需要其他参与者的电子邮件地址,而我是第一个参与者。

这是消息控制器的代码:

[BotAuthentication]
public class MessagesController : ApiController
{
    [HttpPost]
    public async Task<HttpResponseMessage> Post([FromBody] Activity activity)
    {
        using (var connector = new ConnectorClient(new Uri(activity.ServiceUrl)))
        {
            if (activity.IsComposeExtensionQuery())
            {
                // Invoke the command handler
                var response = await MessageExtension.HandleMessageExtensionQuery(connector, activity).ConfigureAwait(false);
                return response != null
                    ? Request.CreateResponse<ComposeExtensionResponse>(response)
                    : new HttpResponseMessage(HttpStatusCode.OK);
            }
            else
            {
                await EchoBot.EchoMessage(connector, activity);
                return new HttpResponseMessage(HttpStatusCode.Accepted);
            }
        }
    }
}

代码来自MessageExtension.HandleMessageExtensionQuery如下:

    public static async Task<ComposeExtensionResponse> HandleMessageExtensionQuery(ConnectorClient connector, Activity activity)
    {
        var query = activity.GetComposeExtensionQueryData();
        if (query == null)
        {
            return null;
        }

        // Exception thrown here - error 403, there is no additional data except "Operation returned an invalid status code 'Forbidden'"
        var members = await connector.Conversations.GetConversationMembersAsync(activity.Conversation.Id);

        var handler = GetCommandHandler(query.CommandId); // Gets a handler based on the command, irrelevant for this question
        if (handler == null)
        {
            return null;
        }

        return await handler.HandleCommand(query, members); // Should handle the command, but never comes here if we are in a 1:1 conversation
    }

致电给GetConversationMembersAsync失败并显示以下消息:Operation returned an invalid status code 'Forbidden'如果该命令是从两个人之间 1:1 的个人对话中调用的。

如果从组通道调用,则呼叫不会失败。

如何获取1:1私人对话的参与者名单?我是否必须通过机器人对我的用户进行身份验证才能执行此操作,或者我是否必须授予我的机器人某些特定权限?我的帐户是否需要具有某些特定权限才能执行此操作?

编辑 - 添加了应用程序清单

{
    "$schema": "https://developer.microsoft.com/en-us/json-schemas/teams/v1.5/MicrosoftTeams.schema.json",
    "manifestVersion": "1.5",
    "version": "1.0.0",
    "id": "1ce95960-0417-4469-ab77-5052758a4e7e",
    "packageName": "com.contoso.helloworld",
    "developer": {
        "name": "Contoso",
        "websiteUrl": "https://8112abe3.ngrok.io",
        "privacyUrl": "https://8112abe3.ngrok.io/privacy-policy",
        "termsOfUseUrl": "https://8112abe3.ngrok.io/terms-service"
    },
    "icons": {
        "color": "color.png",
        "outline": "outline.png"
    },
    "name": {
        "short": "Hello World",
        "full": "Hello World App"
    },
    "description": {
        "short": "Hello World App for Microsoft Teams",
        "full": "This sample app provides a very simple app. You can extend this to add more content and capabilities."
    },
    "accentColor": "#60A18E",
    "configurableTabs": [
        {
            "configurationUrl": "https://526d7c43.ngrok.io/configure",
            "canUpdateConfiguration": true,
            "scopes": [
                "team",
                "groupchat"
            ]
        }
    ],
    "staticTabs": [
        {
            "entityId": "com.contoso.helloworld.hellotab",
            "name": "Hello Tab",
            "contentUrl": "https://8112abe3.ngrok.io/hello",
            "websiteUrl": "https://8112abe3.ngrok.io/hello",
            "scopes": [
                "personal"
            ]
        }
    ],
    "bots": [
        {
            "botId": "bfbcb607-5c29-4438-85a5-15e63fb0b273",
            "scopes": [
                "personal",
                "team",
                "groupchat"
            ],
            "supportsFiles": false,
            "isNotificationOnly": false
        }
    ],
    "composeExtensions": [
        {
            "botId": "bfbcb607-5c29-4438-85a5-15e63fb0b273",
            "canUpdateConfiguration": true,
            "commands": [
                {
                    "id": "getRandomText",
                    "type": "query",
                    "title": "Get random text",
                    "description": "",
                    "initialRun": true,
                    "fetchTask": false,
                    "context": [
                        "commandBox",
                        "compose",
                        "message"
                    ],
                    "parameters": [
                        {
                            "name": "cardTitle",
                            "title": "Subject",
                            "description": "",
                            "inputType": "text"
                        }
                    ]
                }
            ]
        }
    ],
    "permissions": [
        "identity",
        "messageTeamMembers"
    ],
    "validDomains": [
        "8112abe3.ngrok.io"
    ]
}

编辑 2 - 尝试后,基于示例 51 - TeamsMessagingExtensionsAction

按照建议,我尝试了示例 51 名为“TeamsMessagingExtensionsAction” https://github.com/microsoft/BotBuilder-Samples/tree/master/samples/csharp_dotnetcore/51.teams-messaging-extensions-action代码是:

MicrosoftAppCredentials.TrustServiceUrl(turnContext.Activity.ServiceUrl);
var members = (await turnContext.TurnState.Get<IConnectorClient>().Conversations.GetConversationMembersAsync(
    turnContext.Activity.Conversation.Id).ConfigureAwait(false)).ToList();

异常以及堆栈跟踪:

Microsoft.Bot.Schema.ErrorResponseException: Operation returned an invalid status code 'Forbidden'
   at Microsoft.Bot.Connector.Conversations.GetConversationMembersWithHttpMessagesAsync(String conversationId, Dictionary`2 customHeaders, CancellationToken cancellationToken) in d:\a\1\s\libraries\Microsoft.Bot.Connector\Conversations.cs:line 1462
   at Microsoft.BotBuilderSamples.Bots.TeamsMessagingExtensionsActionBot.ShareMessageCommand(ITurnContext`1 turnContext, MessagingExtensionAction action) in D:\Visual Studio Projects\botbuilder-samples\samples\csharp_dotnetcore\51.teams-messaging-extensions-action\Bots\TeamsMessagingExtensionsActionBot.cs:line 68
   at Microsoft.BotBuilderSamples.Bots.TeamsMessagingExtensionsActionBot.OnTeamsMessagingExtensionSubmitActionAsync(ITurnContext`1 turnContext, MessagingExtensionAction action, CancellationToken cancellationToken) in D:\Visual Studio Projects\botbuilder-samples\samples\csharp_dotnetcore\51.teams-messaging-extensions-action\Bots\TeamsMessagingExtensionsActionBot.cs:line 29
   at Microsoft.Bot.Builder.Teams.TeamsActivityHandler.OnTeamsMessagingExtensionSubmitActionDispatchAsync(ITurnContext`1 turnContext, MessagingExtensionAction action, CancellationToken cancellationToken) in d:\a\1\s\libraries\Microsoft.Bot.Builder\Teams\TeamsActivityHandler.cs:line 201
   at Microsoft.Bot.Builder.Teams.TeamsActivityHandler.OnInvokeActivityAsync(ITurnContext`1 turnContext, CancellationToken cancellationToken) in d:\a\1\s\libraries\Microsoft.Bot.Builder\Teams\TeamsActivityHandler.cs:line 88
   at Microsoft.Bot.Builder.Teams.TeamsActivityHandler.OnTurnAsync(ITurnContext turnContext, CancellationToken cancellationToken) in d:\a\1\s\libraries\Microsoft.Bot.Builder\Teams\TeamsActivityHandler.cs:line 39
   at Microsoft.Bot.Builder.BotFrameworkAdapter.TenantIdWorkaroundForTeamsMiddleware.OnTurnAsync(ITurnContext turnContext, NextDelegate next, CancellationToken cancellationToken) in d:\a\1\s\libraries\Microsoft.Bot.Builder\BotFrameworkAdapter.cs:line 1158
   at Microsoft.Bot.Builder.MiddlewareSet.ReceiveActivityWithStatusAsync(ITurnContext turnContext, BotCallbackHandler callback, CancellationToken cancellationToken) in d:\a\1\s\libraries\Microsoft.Bot.Builder\MiddlewareSet.cs:line 55
   at Microsoft.Bot.Builder.BotAdapter.RunPipelineAsync(ITurnContext turnContext, BotCallbackHandler callback, CancellationToken cancellationToken) in d:\a\1\s\libraries\Microsoft.Bot.Builder\BotAdapter.cs:line 182

编辑 3 - 尝试使用示例 57

所以我开始了样本 57 名为“TeamsConversationBot” https://github.com/microsoft/BotBuilder-Samples/tree/master/samples/csharp_dotnetcore/57.teams-conversation-bot和 TLDR;在频道中有效,在私人对话中无效。

以下是执行的步骤:
注意:ngrok 已经使用命令启动ngrok http -host-header=rewrite 3978我现有的机器人注册已配置为侦听给定的 URL

  1. 更新了项目的 appsettings.json 配置,以使用 Bot Framework 注册中的 Microsoft 应用程序 ID 和应用程序密码。
  2. 使用所需的 GUID 编辑 TeamsAppManifest 中包含的 Manifest.json,将其与 PNG 图标一起压缩到 Manifest.zip 中,并使用“上传到 Teams App Studio”导入现有应用程序”,这产生了一个名为“TeamsConversationBot”的新应用程序
  3. 在 Teams App Studio 中,打开新应用程序进行编辑,转到“测试和分发”,点击“Install“ 进而 ”Add“在下一个屏幕上。
  4. 再次导航到上面的屏幕,但不是“Add“ 我选择了 ”添加到团队“从下拉菜单中,我将机器人添加到了我所拥有的团队的频道中。
  5. 在 Visual Studio 中,转到TeamsConversationsBot.cs并设置断点消息所有成员异步方法,在一行中写着var members = await TeamsInfo.GetMembersAsync(turnContext, cancellationToken);
  6. 启动项目,转到我在 Teams 中的频道,发送“给所有会员留言” 对于我的机器人,等待上述断点被击中并观察到上述调用成功,即返回成员列表。到目前为止,一切都很好!
  7. 返回 Teams 应用程序生成器编辑我的“Teams 对话机器人”应用程序。
  8. 在清单编辑器中,转到“消息传递扩展”选项卡并设置我现有的机器人来侦听消息传递扩展调用。
  9. 添加了一个新的基于操作的命令,其中包含已定义的参数数量。给它一个 Id、命名一个参数并输入其他必填字段,这并不重要。对于“选择撰写扩展应该工作的上下文”,我选择了“命令盒" and "撰写框“,勾选”初始运行”并点击“Save".
  10. 使用“再次安装该应用程序测试和分发” -> “Install“ 进而 ”Add“ 按钮。
  11. 在Visual Studio中,添加以下方法TeamsConversationBot.cs:

    protected override async Task<InvokeResponse> OnInvokeActivityAsync(ITurnContext<IInvokeActivity> turnContext, CancellationToken cancellationToken)
    {
        var connector = turnContext.TurnState.Get<IConnectorClient>();
        var conversation = turnContext.Activity.Conversation;
        var members = await connector.Conversations.GetConversationMembersAsync(conversation.Id, cancellationToken);
        return await base.OnInvokeActivityAsync(turnContext, cancellationToken);
    }
    
  12. 在行上设置一个断点var members = await connector.Conversations.GetConversationMembersAsync(conversation.Id, cancellationToken);并启动项目

  13. 进入频道,在撰写框下选择消息传递扩展并调用新添加的命令。这触发了上面的断点。执行一步并观察到members变量包含频道的所有成员。所以这也有效!
  14. 进行我和其他团队成员之间的私人对话(我和他们之前都已互相发送消息,以确保对话不为空),选择撰写框下的消息扩展并调用新添加的命令。这再次触发了上述断点。执行了一步并bam!调用导致了未处理的异常Microsoft.Bot.Schema.ErrorResponseException: Operation returned an invalid status code 'Forbidden'

对编辑 3 的回应

抱歉,看来我误解了您想要做什么。您无法使用 Bot Framework 执行此操作。从技术上讲,机器人不是您和其他用户之间个人对话的一部分。因此,它无权获取有关该对话的信息,即使它可作为消息传递扩展使用。

您也许可以使用,尽管您必须自己处理获取身份验证令牌的问题:

https://graph.microsoft.com/beta/chats/<conversationId>/members

将此作为答案,因为评论太长,我认为它应该满足您的需求。请告诉我这是否有效,我可以编辑它


通过查看您的内容,我没有在后端看到任何内容appId,所以我不确定问题的实际原因是什么。

您根本不需要设置任何权限——这些权限仅适用于 OAuth,这GetConversationMembersAsync不需要。您应该只真正需要在您的manifest.json,看起来设置得很好。

但是,对于您的应用程序注册,您可能需要确保在身份验证下选中此项:

如果不是,您可以手动调整应用程序注册清单,通过设置:

"signInAudience": "AzureADandPersonalMicrosoftAccount",

如果这些都不起作用:

请尝试样品57 https://github.com/microsoft/BotBuilder-Samples/tree/master/samples/csharp_dotnetcore/57.teams-conversation-bot,其中有一个GetMembersAsync https://github.com/microsoft/BotBuilder-Samples/blob/master/samples/csharp_dotnetcore/57.teams-conversation-bot/Bots/TeamsConversationBot.cs#L101函数可以完成同样的事情,但是使用当前的 BotFramework SDK。请务必遵循自述文件并按指定设置所有内容。

让我知道事情的后续。如果您需要其他帮助,您可以尝试尽可能详细地解释您的复制步骤吗?我想尝试重现这一点,并希望确保我们的步骤匹配——它也可以帮助我找出您可能出错的地方。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

调用 GetConversationMembersAsync 进行 1:1 私人对话时,操作返回无效状态代码“禁止” 的相关文章

随机推荐