我已经在C#中为多租户Azure机器人实现了以下功能,这对于验证应用程序凭据非常有用。
C# : 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表存储中?
答案 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
并使用正确的连接字符串。