Внедрение зависимостей IoC в пользовательский HTTP-модуль - как? (ASP.NET)

У меня есть собственный HTTP-модуль. Я хотел бы ввести регистратор, используя мою инфраструктуру IoC, поэтому я могу регистрировать ошибки в модуле. Однако, конечно, я не получаю конструктор, поэтому не могу его вставить. Какой лучший способ сделать это?

Если вам нужен конкретный контейнер IoC - я в настоящее время использую Windsor, но могу скоро перейти в AutoFac.

Спасибо

Ответ 2

В первый раз я увидел инъекцию зависимостей в HttpModules в Spring.NET(но не рекламу этой среды). Идея состоит в том, что у вас есть специальный HttpModule, который вводит зависимости другим HttpModule уровня приложения.

Несомненно, текущая версия Autofac.Integration.Web не поддерживает это, но вы можете легко сделать это сами:

public class MyModule : IHttpModule
{
    public void Dispose()
    {            
    }

    public void Init(HttpApplication context)
    {
        Assert.IsNotNull(MyService);
    }        

    public IMyService MyService { get; set; }
}

public class HttpModuleInjectionModule : IHttpModule
{
    public void Dispose()
    {            
    }

    public void Init(HttpApplication context)
    {
        var containerProviderAccessor = context as IContainerProviderAccessor;

        if(containerProviderAccessor == null)
            throw new InvalidOperationException("HttpApplication should implement IContainerProviderAccessor");

        var rootContainer = containerProviderAccessor.ContainerProvider.ApplicationContainer;

        foreach (string moduleName in context.Modules.AllKeys)
            rootContainer.InjectProperties(context.Modules[moduleName]);
    }
}

public class Global : HttpApplication, IContainerProviderAccessor
{
  static IContainerProvider _containerProvider;

  protected void Application_Start(object sender, EventArgs e)
  {
    var builder = new ContainerBuilder();
    builder.Register<MyService>().As<IMyService>();
    _containerProvider = new ContainerProvider(builder.Build());
  }

  public IContainerProvider ContainerProvider
  {
    get { return _containerProvider; }
  }
}

HttpModuleInjectionModule должен быть зарегистрирован перед другими HttpModule-s в web.config:

  <httpModules>
   <add name="ScriptModule" type="System.Web.Handlers.ScriptModule, System.Web.Extensions, Version=3.5.0.0, Culture=neutral, PublicKeyToken=31BF3856AD364E35"/>
   <add name="HttpModuleInjection" type="WebTest.HttpModuleInjectionModule, WebTest"/>
   <add name="ContainerDisposal" type="Autofac.Integration.Web.ContainerDisposalModule, Autofac.Integration.Web"/>
   <add name="PropertyInjection" type="Autofac.Integration.Web.PropertyInjectionModule, Autofac.Integration.Web"/>
   <add name="MyModule" type="WebTest.MyModule, WebTest"/>
  </httpModules>

Я уверен, что вы можете делать подобные вещи в Виндзоре. Разница заключается в том, как вы получаете доступ к вашему корневому контейнеру из HttpModuleInjectionModule.

Ответ 3

Вы можете передать необходимые зависимости через контекст HttpApplication, переданный вам методом Init...

public class MyHttpModule : IHttpModule
{
    public void Init(HttpApplication context)
    {
        var dependency = (IDependency)context.Context.Items["dependency"];
        // consume dependency...
    }

    public void Dispose()
    {
    }
}

Ответ 4

Мне любопытно ответить на андре-цыкунов, но у меня нет комментариев, чтобы прокомментировать это.

Я пытаюсь получить удобство с IoC и DI, поэтому мне может что-то не хватает, но не проще ли использовать IContainerProviderAccessor из MyModule, а не создавать другой модуль?

Например:

public class MyModule : IHttpModule
{
    public void Dispose()
    {            
    }

    public void Init(HttpApplication context)
    {
        Assert.IsNotNull(MyService);

        var containerProviderAccessor = context as IContainerProviderAccessor;

        if (accessor != null)
        {
            IContainer container = containerProviderAccessor.ContainerProvider.ApplicationContainer;
            MyService = container.Resolve<IMyService>();
        }
    }

    private IMyService MyService { get; set; }
}