多租户机器人数据存储

时间:2018-08-26 02:05:05

标签: c# botframework multi-tenant azure-table-storage

我已经在C#中为多租户Azure机器人实现了以下功能,这对于验证应用程序凭据非常有用。

C# : Creating a Single Bot Service to Support Multiple Bot Applications

https://www.microsoft.com/developerblog/2017/01/10/creating-a-single-bot-service-to-support-multiple-bot-applications/

但是,我想知道机器人数据存储区是否也可以做到这一点? 我目前已经实现了标准的Azure表存储bot数据存储,就像这样,它使用web.config中定义的单个连接字符串:

Conversation.UpdateContainer(
            builder =>
            {
                builder.RegisterModule(new ReflectionSurrogateModule());

                builder.RegisterModule(new AzureModule(Assembly.GetExecutingAssembly()));

                //var store = new InMemoryDataStore();

                var store = new TableBotDataStore(ConfigurationManager.AppSettings["BotStorage"]);

                builder.Register(c => store)
                    .Keyed<IBotDataStore<BotData>>(AzureModule.Key_DataStore)
                    .AsSelf()
                    .SingleInstance();
            });

在我的ICredentialProvider实现中(如文章示例中所述)。如果我将客户连接字符串与他们的应用程序ID和密码一起存储,则可以根据客户/机器人对数据存储进行序列化,在上面的依赖项注入中提取该引用,然后将数据保存/读取到客户Azure表存储中?

1 个答案:

答案 0 :(得分:1)

您将需要创建一个实现IBotDataStore<BotData>的自定义漫游器数据存储并使用它。我能够得到一个简单的每次创建相同的TableBotDataStore的脚本。以下是我使用的方法,每次发送消息时,构造函数中的断点都会被命中。

public class CustomBotStore : IBotDataStore<BotData>
{
    private IBotDataStore<BotData> _store;

    public CustomBotStore()
    {
        string connString = ConfigurationManager.ConnectionStrings["BotState"].ConnectionString;

        _store = new TableBotDataStore(connString, "testbotdata"); // requires Microsoft.BotBuilder.Azure Nuget package 
    }

    public Task<bool> FlushAsync(IAddress key, CancellationToken cancellationToken)
    {
        return _store.FlushAsync(key, cancellationToken);
    }

    public Task<BotData> LoadAsync(IAddress key, BotStoreType botStoreType, CancellationToken cancellationToken)
    {
        return _store.LoadAsync(key, botStoreType, cancellationToken);
    }

    public Task SaveAsync(IAddress key, BotStoreType botStoreType, BotData data, CancellationToken cancellationToken)
    {
        return _store.SaveAsync(key, botStoreType, data, cancellationToken);
    }
}


public class WebApiApplication : System.Web.HttpApplication
{
    protected void Application_Start()
    {
        GlobalConfiguration.Configure(WebApiConfig.Register);

        Conversation.UpdateContainer(
        builder =>
        {
            builder.RegisterModule(new AzureModule(Assembly.GetExecutingAssembly()));

            builder.Register(c => new CustomBotStore())
                .Keyed<IBotDataStore<BotData>>(AzureModule.Key_DataStore)
                .AsSelf()
                .InstancePerLifetimeScope();
        });
    }
}

我不确定每个生命周期范围内拥有一个实例的含义是什么,因为该示例使用单个实例。但是您应该可以使用它,然后在构造函数中检查HttpContext并使用正确的连接字符串。