Azure 聊天机器人中间件的依赖注入?

Dependency Injection for Azure chat bot middleware?

我正在使用 Azure 机器人服务和 QnAMaker 开发一个新的聊天机器人。我们正在使用 BotBuilder 中间件(包括自定义中间件)来定制机器人行为。

其中一个中间件将调用 Azure 函数,我想将新的 HttpClientFactory 功能与自定义中间件一起使用 - 但这需要依赖项注入。

如何像使用常规 .NET Core 中间件一样在 BotBuilder 中间件中使用依赖注入?

当您查看 Startup.cs 中的机器人配置时,您会发现它如何要求您 new 添加所有机器人依赖项:

services.AddHttpClient<MyFunctionClient>(client =>
{
    client.BaseAddress = new Uri(mySettings.GetValue<string>("myFunctionUrl"));
    client.DefaultRequestHeaders.Add("x-functions-key", mySettings.GetValue<string>("myFunctionKey"));
});

services.AddBot<QnAMakerBot>(options =>
{  
    options.CredentialProvider = new ConfigurationCredentialProvider(Configuration);

    options.ConnectorClientRetryPolicy = new RetryPolicy(
        new BotFrameworkHttpStatusCodeErrorDetectionStrategy(),
        3,
        TimeSpan.FromSeconds(2), 
        TimeSpan.FromSeconds(20),
        TimeSpan.FromSeconds(1));

    var middleware = options.Middleware;
    middleware.Add(new ConversationState<ChatLog>(new MemoryStorage()));
    middleware.Add(new MyCustomMiddleware()); // <- I want to inject a typed HttpClient here

//... etc. ....

是否有不同的方法来配置允许依赖项注入的机器人?

如果 MyCustomMiddleware 在其构造函数中需要类型化 HttpClient,我必须在此处创建一个新实例,因此我无法获得 DI 和我刚刚设置的配置的好处向上。

虽然我不是服务定位器模式的粉丝,但 bot 配置的当前设计并不是非常依赖注入友好。

利用机器人中间件的设置方式的性质,但必须在启动期间提供一个新实例,我想出了以下解决方法。

public class BotMiddlewareAdapter<TMiddleware> : IMiddleware
    where TMiddleware : IMiddleware {
    private readonly Lazy<TMiddleware> middleware;

    public BotMiddlewareAdapter(IServiceCollection services) {
        middleware = new Lazy<TMiddleware>(() =>
            services.BuildServiceProvider().GetRequiredService<TMiddleware>());
    }

    public Task OnTurn(ITurnContext context, MiddlewareSet.NextDelegate next) {
        return middleware.Value.OnTurn(context, next);
    }
}

它将 IServiceCollection 作为显式依赖项,并推迟了服务提供者的创建和工厂委托中实际中间件的最终解析。

然后可以使用

实现
middleware.Add(new BotMiddlewareAdapter<MyCustomMiddleware>(services));

调用适配器时,它会在初始调用时延迟解析预期的中间件,然后调用它。

事实上,您可以更进一步并将其转换为扩展方法

public static class BotBuilderMiddlewareExtension {
    public static void Add<TMiddleware>(this IList<IMiddleware> middleware, IServiceCollection services) 
        where TMiddleware : IMiddleware {
        middleware.Add(new BotMiddlewareAdapter<TMiddleware>(services));
    }
}

这将设置简化为

middleware.Add<MyCustomMiddleware>(services);