MASA Framework -- EventBus入门与设计( 二 )


  1. 发送注册用户事件,修改Program.cs
app.MapPost("/register", async (RegisterUserEvent @event, IEventBus eventBus) =>{    await eventBus.PublishAsync(@event);});进阶处理流程EventBus的 请求管道包含一系列请求委托,依次调用 。它们与ASP.NET Core中间件有异曲同工之妙,区别点在于中间件的执行顺序与注册顺序相反,最先注册的最后执行
MASA Framework -- EventBus入门与设计

文章插图
每个委托均可在下一个委托前后执行操作,其中TransactionMiddleware是EventBus发布后第一个要进入的中间件 (默认提供),并且它是不支持多次嵌套的 。
EventBus 支持嵌套,这意味着我们可以在Handler中重新发布一个新的Event,但TransactionMiddleware仅会在最外层进入时被触发一次
自定义中间件根据需要我们可以自定义中间件,并注册到EventBus的请求管道中,比如通过增加FluentValidation, 将参数验证从业务代码中剥离开来,从而使得处理程序更专注于业务
  1. 注册FluentValidation, 修改Program.cs
builder.Services.AddValidatorsFromAssembly(Assembly.GetEntryAssembly());
  1. 自定义验证中间件ValidatorMiddleware.cs,用于验证参数
public class ValidatorMiddleware<TEvent> : Middleware<TEvent>    where TEvent : IEvent{    private readonly ILogger<ValidatorMiddleware<TEvent>>? _logger;    private readonly IEnumerable<IValidator<TEvent>> _validators;    public ValidatorMiddleware(IEnumerable<IValidator<TEvent>> validators, ILogger<ValidatorMiddleware<TEvent>>? logger = null)    {        _validators = validators;        _logger = logger;    }    public override async Task HandleAsync(TEvent @event, EventHandlerDelegate next)    {        var typeName = @event.GetType().FullName;        _logger?.LogDebug("----- Validating command {CommandType}", typeName);        var failures = _validators            .Select(v => v.Validate(@event))            .SelectMany(result => result.Errors)            .Where(error => error != null)            .ToList();        if (failures.Any())        {            _logger?.LogError("Validation errors - {CommandType} - Event: {@Command} - Errors: {@ValidationErrors}",                typeName,                @event,                failures);            throw new ValidationException("Validation exception", failures);        }        await next();    }}

经验总结扩展阅读