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,string>), 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?
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();
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});