web-dev-qa-db-fra.com

Ajouter une validation à un pipeline de comportements MediatR?

J'utilise ASP.NET Core, le conteneur intégré et MediatR 3 qui prend en charge pipelines "comportement" :

public class MyRequest : IRequest<string>
{
    // ...
}

public class MyRequestHandler : IRequestHandler<MyRequest, string>
{
    public string Handle(MyRequest message)
    {
        return "Hello!";
    }
}

public class MyPipeline<TRequest, TResponse> : IPipelineBehavior<TRequest, TResponse>
{
    public async Task<TResponse> Handle(TRequest request, RequestHandlerDelegate<TResponse> next)
    {
        var response = await next();
        return response;
    }
}

// in `Startup.ConfigureServices()`:
services.AddTransient(typeof(IPipelineBehavior<MyRequest,str‌​ing>), typeof(MyPipeline<MyRequest,string>))

J'ai besoin d'un validateur FluentValidation dans le pipeline. Dans MediatR 2, un le pipeline de validation a été créé ainsi :

public class ValidationPipeline<TRequest, TResponse>
    : IRequestHandler<TRequest, TResponse>
    where TRequest : IRequest<TResponse>
{

    public ValidationPipeline(IRequestHandler<TRequest, TResponse> inner, IEnumerable<IValidator<TRequest>> validators)
    {
        _inner = inner;
        _validators = validators;
    }

    public TResponse Handle(TRequest message)
    {
        var failures = _validators
            .Select(v => v.Validate(message))
            .SelectMany(result => result.Errors)
            .Where(f => f != null)
            .ToList();
        if (failures.Any())
            throw new ValidationException(failures);
        return _inner.Handle(request);
    }

}

Comment faire maintenant pour la nouvelle version? Comment définir quel valideur utiliser?

19
grokky

Le processus est exactement le même, il suffit de changer l'interface pour utiliser le nouveau IPipelineBehavior<TRequest, TResponse> interface.

public class ValidationBehavior<TRequest, TResponse> : IPipelineBehavior<TRequest, TResponse>
    where TRequest : IRequest<TResponse>
{
    private readonly IEnumerable<IValidator<TRequest>> _validators;

    public ValidationBehavior(IEnumerable<IValidator<TRequest>> validators)
    {
        _validators = validators;
    }

    public Task<TResponse> Handle(TRequest request, RequestHandlerDelegate<TResponse> next)
    {
        var context = new ValidationContext(request);
        var failures = _validators
            .Select(v => v.Validate(context))
            .SelectMany(result => result.Errors)
            .Where(f => f != null)
            .ToList();

        if (failures.Count != 0)
        {
            throw new ValidationException(failures);
        }

        return next();
    }
}

Pour les validateurs, vous devez enregistrer tous les validateurs en tant que IValidator<TRequest> dans le conteneur intégré afin qu'ils soient injectés dans le comportement. Si vous ne voulez pas les enregistrer un par un, je vous suggère de jeter un œil à la grande bibliothèque Scrutor qui apporte des capacités de numérisation d'assemblage. De cette façon, il trouvera vos validateurs lui-même.

De plus, avec le nouveau système, vous n'utilisez plus le motif de décoration, vous enregistrez simplement votre comportement générique dans le conteneur et MediatR le récupérera automatiquement. Cela pourrait ressembler à quelque chose comme:

var services = new ServiceCollection();
services.AddMediatR(typeof(Program));
services.AddTransient(typeof(IPipelineBehavior<,>), typeof(ValidationBehavior<,>));
var provider = services.BuildServiceProvider();
17
Mickaël Derriey

J'ai intégré l'intégration du noyau .net dans nuget, n'hésitez pas à l'utiliser: https://www.nuget.org/packages/MediatR.Extensions.FluentValidation.AspNetCore

Insérez simplement dans la section de configuration:

services.AddFluentValidation(new[] {typeof(GenerateInvoiceHandler).GetTypeInfo().Assembly});

GitHub

0
GetoX