Ninject WebAPI由于已经处理了DbContext,因此无法完成操作

时间:2013-03-11 21:30:21

标签: entity-framework asp.net-web-api repository ninject

所以我使用了一个带有属性和过滤器的简单存储库模式,因为我使用的是来自NuGet的Ninject.Web.WebApi-RC软件包。{/ 3}

这适用于第一个请求,但由于我在请求范围内有DbContext,因此会在所有后续请求中处理。

这是我的属性:

public class CommunicationKeyValidationAttribute : FilterAttribute
{
}

这是我的过滤器:

public class CommunicationKeyValidationFilter : AbstractActionFilter
{
    public CommunicationKeyValidationFilter(IRepository repository)
    {
        this.repository = repository;
    }
    public override void OnActionExecuting(HttpActionContext actionContext)
    {
        // do stuff
    }
 }

这是我的存储库:

public class Repository : IRepository
{
    public Repository(MyDbContext dbContext)
    {
        this.dbContext = dbContext;
    }
}

以下是我的Ninject绑定:

this.Kernel.Bind<MyDbContext>().ToSelf().InRequestScope();
this.Kernel.Bind<IRepository>().To<Repository>().InRequestScope();
this.Kernel.BindHttpFilter<CommunicationKeyValidationFilter>(FilterScope.Action)
        .WhenActionMethodHas<CommunicationKeyValidationAttribute>()
        .InRequestScope();

我的控制器看起来像这样:

public class HomeController 
{
    [CommunicationKeyValidation]
    public ActionResult Index()
    {
        // do stuff
    }

这里的问题是CommunicationKeyValidationFilter上的构造函数仅在第一个请求时被调用。有没有办法可以在每次尝试解析过滤器时让ninject构造这个过滤器?

1 个答案:

答案 0 :(得分:15)

过滤器由WebApi缓存。它们应该处于瞬态范围内,以便WebApi可以管理生命周期。由于生命周期较长,因此您不能拥有任何生命周期较短的依赖项。

您可以做的是在执行过滤器期间创建存储库。为此,最好使用NinjectFactoryExtension注入工厂:

public class CommunicationKeyValidationFilter : AbstractActionFilter
{
    public CommunicationKeyValidationFilter(IRepositoryFactory repositoryFactory)
    {
        this.repositoryFactory = repositoryFactory;
    }

    public override void OnActionExecuting(HttpActionContext actionContext)
    {
        var repository = this.repositoryFactory.CreateRepository();
    }
 }

 public interface IRepositoryFactory { IRepository CreateRepository(); }
 kernel.Bind<IRepositoryFactory>().ToFactory();