事件总线解决了微服务间如何基于集成事件进行异步通讯的问题。然而只有事件总线正常运行,微服务之间基于事件的通讯才得以运转。
而现实状况是,总有这样或那样的问题,致使事件总线不稳定或不可用,好比:网络中断,系统断电等等,这均可能致使微服务间的不一致性问题。
那如何解决事件总线故障致使的不一致问题呢?sql
发件箱模式数据库
var oldPrice = item.Price; item.Price = product.Price; _context.CatalogItems.Update(item); var @event = new ProductPriceChangedIntegrationEvent(item.Id, item.Price, oldPrice); // Commit changes in original transaction await _context.SaveChangesAsync(); // Publish integration event to the event bus // (RabbitMQ or a service bus underneath) _eventBus.Publish(@event);
当产品价格更改后,代码将数据提交给数据库,而后发布ProductPriceChangedIntegrationEvent
事件。
若是服务在数据库更新后崩溃(奔溃发生在_context.SaveChangesAsync()
代码执行以后,但又发生在集成事件成功发布前),就会致使本地微服务价格已成功更新,但集成事件未发布的问题。就会致使目录微服务中定义的价格和顾客购物车中缓存的价格不一致。缓存
以上问题的关键在因而如何确保两个独立的操做的原子性。若是单从单体应用的角度来处理的话,咱们彻底是能够将他们放到同一个事务中去保证。然而在微服务中,就违背了其高可用的基本要求。由于一旦事件总线处于瘫痪状态,那么整个目录微服务就不可用了。这种强制经过事务保证的一致性,就引入了太多的问题依赖。网络
若是从微服务的角度来看,每一个微服务负责各自的业务逻辑,对于目录微服务来讲,它的关注点是产品的更新是否成功。至于借助事件总线经过异步事件实现微服务间的通讯,并非其关注点。这也就是关注点分离。换句话说,产品的更新不该该依赖外部状态。在这里,外部状态就是事件总线的可用性。异步
你可能会说了,既然不容许经过强事务保证一致性,那么如何解决一致性问题呢(好像绕了半天又回到了原点)?微服务
这里就要引入强一致性和最终一致性的概念了。
强一致性:也就是事务一致性,将多个操做放到单一事务处理。要么所有成功,要么所有失败。
最终一致性:经过将某些操做的执行延迟到稍后的时间来执行。若前面的操做执行成功,后续操做将延后执行。若前面的操做失败,后续的操做就不会执行。
线程
到这里,咱们实际要解决的问题就明确了:如何确保事件总线可以正确进行事件转发?日志
换句话说:事件总线挂了,可是事件消息不能丢失。只要事件消息不丢,后面咱们还有机会挽救(从新发布消息)。code
如何保证事件消息不丢失呢?固然是持久化了。blog
eShopOnContainers已经考虑了这一点,集成了事件日志用于持久化。咱们直接来看类图:
从类图中看其实现逻辑也很简单,主要是定义了一个IntegrationEventLogEntry
实体、EventStateEnum
事件状态枚举和IntegrationEventLogContext
EF上下文用于事件日志持久化。暴露IIntegrationEventLogService
用于事件状态的更新。
其余微服务经过在启动类中注册IntegrationEventLogContext
便可完成事件日志的集成。
services.AddDbContext<IntegrationEventLogContext>(options => { options.UseSqlServer(configuration["ConnectionString"], sqlServerOptionsAction: sqlOptions => { sqlOptions.MigrationsAssembly(typeof(Startup) .GetTypeInfo().Assembly.GetName().Name); sqlOptions.EnableRetryOnFailure(maxRetryCount: 10, maxRetryDelay: TimeSpan.FromSeconds(30), errorNumbersToAdd: null); }); });
使用EF进行数据库迁移后,就会生成IntergrationEventLog
表。以下图所示:
主要分两步走:
第一步毋庸置疑,第二步发布事件,咱们又有多种实现方式:
这里很显然第二种方式更为稳妥。而eShopOnContainers出于简单考虑,采用了第一种方案,具体代码以下:
using (var transaction = _catalogContext.Database.BeginTransaction()) { _catalogContext.CatalogItems.Update(catalogItem); await _catalogContext.SaveChangesAsync(); // Save to EventLog only if product price changed if(raiseProductPriceChangedEvent) await _integrationEventLogService.SaveEventAsync(priceChangedEvent); transaction.Commit(); } // Publish the intergation event through the event bus _eventBus.Publish(priceChangedEvent); integrationEventLogService.MarkEventAsPublishedAsync( priceChangedEvent);
至此,eShopOnContainers确保事件总线可以正确转发消息的解决方案阐述完毕。你可能会问,这对应的是引言中的哪种方案?都不是,你能够看做其是基于事件日志的简化版的事件溯源。
经过持久化事件日志来避免事件发布失败致使的一致性问题,是一种有效措施。然而消息从发送到接收再到正常消费的过程当中,每个环节均可能故障,因此仅仅在消息发送端使用事件日志只是确保最终一致性的一小步。还有不少问题有待完善:
而这些问题就留给你们思考吧。