Autofac/FluentValidation:没有标签匹配的作用域';AutofacWebRequest

本文关键字:作用域 AutofacWebRequest FluentValidation 标签 Autofac | 更新日期: 2023-09-27 18:26:12

尝试将数据注入FluentValidation验证器:

public class MyFormValidator : AbstractValidator<MyForm>
{
    private readonly IQueryable<Models.User> _users;
    public MyFormValidator(IQueryable<Models.User> users)
    {
        _users = users;
        ...
    }
}

我的验证器工厂:

public class DependencyResolverValidatorFactory : ValidatorFactoryBase
{
    private readonly IContainer container;
    public DependencyResolverValidatorFactory(IContainer container)
    {
        this.container = container;
    }
    public override IValidator CreateInstance(Type validatorType)
    {
        return container.ResolveOptionalKeyed<IValidator>(validatorType);
    }
}

我的Autofac配置程序:

public class AutofacConfigurator
{
    public static void Configure()
    {
        var builder = new ContainerBuilder();
        ...
        builder.RegisterType<MyFormValidator>()
            .Keyed<IValidator>(typeof(IValidator<MyForm>))
            .As<IValidator>()
             // 2nd parameter returns IQueryable<User>
            .WithParameter("users", new SqlRepository<User>(dataContext)) 
            .InstancePerRequest();
        builder.RegisterSource(new AnyConcreteTypeNotAlreadyRegisteredSource());
        var container = builder.Build();
        DependencyResolver.SetResolver(new AutofacDependencyResolver(container));
        // Register the validator factory with FluentValidation, and register 
        // FluentValidation as the model validator provider for the MVC framework. 
        // see http://www.jerriepelser.com/blog/using-fluent-validation-with-asp-net-mvc-part-3-adding-dependency-injection
        var fluentValidationModelValidatorProvider = 
            new FluentValidationModelValidatorProvider(
                new DependencyResolverValidatorFactory(container));
        DataAnnotationsModelValidatorProvider.AddImplicitRequiredAttributeForValueTypes = false;
        fluentValidationModelValidatorProvider.AddImplicitRequiredValidator = false;
        ModelValidatorProviders.Providers.Add(fluentValidationModelValidatorProvider);
    }
}

获得以下异常:

从请求实例的作用域中看不到标记与"AutofacWebRequest"匹配的作用域。这通常表示根据HTTP请求注册的组件正由SingleInstance()组件请求(或类似场景)。在web集成下,始终从DependencyResolver.Current或ILifetimeScopeProvider.RequestLifetime请求依赖项,而不是从容器本身请求。

我还有其他验证器,其中大多数验证器不需要注入数据。

这对我来说是一个很大的新领域(在Autofac和FluentValidation方面),我仍在努力理解我在这里做什么。我怀疑我只是错误地注册了我的类型。如何解决此问题并正确注册我的类型?

(如果这与已经提出的其他问题过于相似,我深表歉意。)

Autofac/FluentValidation:没有标签匹配的作用域';AutofacWebRequest

我对FluentValidation没有任何经验,但我怀疑这是否是您问题的原因,所以我无论如何都会继续努力。

您得到的异常意味着Autofac无法将您的服务解析为"每个请求的实例"。Autofac文档页面上有很多关于这意味着什么的文档。总之,这意味着Autofac将尝试从为发送到Web服务器的每个请求自动创建的生存期范围中解析服务。当您将某个服务注册为.InstancePerRequestScope(),然后尝试在该范围之外解析该服务时,您将得到您看到的DependencyResolutionException

因此,我们已经确定您的MyFormValidator不是从"请求"范围中解决的。为什么?

您编写的自定义DependencyResolverValidatorFactory采用Autofac构建的实际IContainer,并从中解析。这是ILifetimeScope的一种特殊类型,即"根作用域"。没有与此直接相关的请求生存期范围,所以您会得到异常。您需要从从"request"范围开始的ILifetimeScope或包含在请求范围内的子范围进行解析。

Autofac/MVC集成已经自动承载了一个请求范围(在AutofacDependencyResolver中,请参阅源代码),但您的自定义DependencyResolverValidatorFactory无法从中解析。如果您想这样做,我想您可以修改DependencyResolverValidatorFactory以接受AutofacDependencyResolver实例,并使用它来解析。

它看起来像这样:

public class DependencyResolverValidatorFactory : ValidatorFactoryBase
{
    private readonly AutofacDependencyResolver resolver;
    public DependencyResolverValidatorFactory(AutofacDependencyResolver resolver)
    {
        this.resolver = resolver;
    }
    public override IValidator CreateInstance(Type validatorType)
    {
        return resolver.RequestLiftimeScope.ResolveOptionalKeyed<IValidator>(validatorType);
    }
}

注意RequestLifetimeScope卡在那里。然后使用在.Configure()方法中创建

var resolver = new AutofacDependencyResolver(container);
DependencyResolver.SetResolver(resolver);
var fluentValidationModelValidatorProvider = 
    new FluentValidationModelValidatorProvider(
        new DependencyResolverValidatorFactory(resolver));

这应该消除了异常,假设这个工厂在创建IValidator s的实例时确实有一个工作请求。如果没有,您可能需要使用默认行为(.InstancePerDependency(),每次请求时都会创建一个新实例)或单例(.SingleInstance())进行注册,这取决于验证器可以或应该如何共享。

祝你好运。