如何使用Autofac来解析Nancy创建的子生命周期范围中的类型的每个请求依赖项的实例

Hug*_*cey 4 c# dependency-injection autofac nancy

我们在Windows服务中托管了多个应用程序,这些应用程序自托管Nancy端点,以便公开有关应用程序操作的检测.

我们使用Autofac作为我们的IOC.几个存储库在所有应用程序共享的核心DLL中注册到根容器中; 然后使用从中导出的引导程序将此容器作为其容器传递给Nancy Nancy.Autofac.Bootstrapper.

我们发现,当Nancy收到Web请求时,它会从根容器中解析对存储库的请求,这导致内存被非垃圾收集器消耗,IDisposable因为根容器不会超出范围(它具有Windows服务的生命周期).这导致服务"泄漏"内存.

然后我们切换到一个模型,我们在Nancy bootstrapper中使用InstancePerRequest重写ConfigureRequestContainer()方法添加了存储库的注册:

protected override void ConfigureRequestContainer(ILifetimeScope container, NancyContext context)
{
    base.ConfigureRequestContainer(container, context);
    PerRequestContainerBuilder().Update(container.ComponentRegistry);
}

private static ContainerBuilder PerRequestContainerBuilder()
{
    var builder = new ContainerBuilder();

    // Dependency for repository
    builder.RegisterType<SystemDateTimeProvider>().InstancePerRequest().As<IDateTimeProvider>();

    // Repository
    builder.RegisterType<BookmarkRepository>().InstancePerRequest().As<IBookmarkRepository>();

    return builder;
}
Run Code Online (Sandbox Code Playgroud)

我们还覆盖了CreateRequestContainer()使用标记创建请求容器的方法MatchingScopeLifetimeTags.RequestLifetimeScopeTag.

protected override ILifetimeScope CreateRequestContainer(NancyContext context)
{
     return ApplicationContainer.BeginLifetimeScope(MatchingScopeLifetimeTags.RequestLifetimeScopeTag);
}
Run Code Online (Sandbox Code Playgroud)

这似乎已经解决了IDisposable没有被处理的问题- 子请求容器被放置在Web请求管道的末端,并且由它解析的对象也被处理并最终被垃圾收集.

我们的问题是,这似乎是将存储库的实现细节泄漏到服务中,因为我们不仅需要注册存储库,ConfigureRequestContainer()还要注册存储库所需的任何其他对象,即如果我们想要更改存储库的实现,我们必须"遍历依赖链"以使用它来注册每个服务中的所需对象 - 这似乎是错误的.

有没有办法让我们可以让Autofac从根容器中解析存储库的支持对象,但是将注册信息保存在Web请求容器的范围内?或者有没有办法在创建根容器时自动将现有注册从子容器复制到子容器中?

khe*_*ang 6

Autofac应自动解析"父"生命周期中的实例.如果使用配置注册InstancePerRequest,Autofac将使用特殊的生命周期标记注册这些服务MatchingScopeLifetimeTags.RequestLifetimeScopeTag,以便以后可以在正确的范围内解析.

这意味着不需要使用Nancy bootstrapper的ConfigureRequestContainer方法来执行请求范围的注册.你已经做到了!只要Nancy使用相同的标签创建请求生命周期InstancePerRequest(默认情况下从Nancy 1.1开始),就应该正确解析服务.

例:

public class Startup
{
    public void Configuration(IAppBuilder app)
    {
        var builder = new ContainerBuilder();

        // Do request-scoped registrations using InstancePerRequest...

        var container = builder.Build();

        // Pass the pre-built container to the bootstrapper
        var bootstrapper = new MyAwesomeNancyBootstrapper(container);

        app.UseNancy(options => options.Bootstrapper = bootstrapper);
    }
}
Run Code Online (Sandbox Code Playgroud)
public class MyAwesomeNancyBootstrapper : AutofacNancyBootstrapper
{
    private readonly ILifetimeScope _lifetimeScope;

    public MyAwesomeNancyBootstrapper(ILifetimeScope lifetimeScope)
    {
        _lifetimeScope = lifetimeScope;
    }

    protected override ILifetimeScope GetApplicationContainer()
    {
        return _lifetimeScope; // Tell Nancy you've got a container ready to go ;)
    }
}
Run Code Online (Sandbox Code Playgroud)

此设置应该足够(从Nancy 1.1开始.在早期版本中,您还必须覆盖该CreateRequestContainer方法并在创建请求生存期范围时传递请求生命周期标记).

编辑:我在https://github.com/khellang/Nancy.AutofacExample为您整理了一个示例