从eventstore重播特定类型的事件

时间:2018-01-30 05:17:29

标签: event-sourcing get-event-store

我目前正在使用Event Store来处理我的活动。我目前需要重播特定类型的事件,因为我已经对订阅和写入数据库的方式进行了更改。

这可能吗?如果是这样,怎么办呢?感谢。

2 个答案:

答案 0 :(得分:0)

您不能告诉EventStore将特定事件重播到持久订阅上,因为持久订阅的目的是保持订阅者的状态。

要实现这种修复,您确实需要一个赶超应用程序来完成工作。

真的,如果您考虑过,如果将所有事件重播到新数据库中,那么那里是否有正确的数据?

所以我有一个控制台应用程序,该应用程序重复使用与持久连接相同的逻辑,但是唯一的区别是:

  1. 我更改了目标数据库连接字符串-因此,这将是一个新的数据库或集合(而不是损坏的数据库)
  2. 它连接到EventStore并从头开始重播所有事件
  3. 它将整个数据库重建到正确的状态
  4. 将业务切换到新数据库

这是EventStore的重点-您只需重播所有事件即可随时建立任何数据库,这将是正确的

您的持久连接处理新的传入事件并应用更新。

答案 1 :(得分:0)

如果启用$ by_event_type投影,则可以在以下位置访问该投影流

/ streams / $ et- {event-type}

https://eventstore.org/docs/projections/system-projections/index.html

然后,您可以根据需要使用.net api进行阅读。

这里有一些代码可以帮助您入门

      private static T GetInstanceOfEvent<T>(ResolvedEvent resolvedEvent) where T : BaseEvent
            {

                var metadataString = Encoding.UTF8.GetString(resolvedEvent.Event.Metadata);
                var eventClrTypeName = JObject.Parse(metadataString).Property(EventClrTypeHeader).Value;
                var @event = JsonConvert.DeserializeObject(Encoding.UTF8.GetString(resolvedEvent.Event.Data), Type.GetType((string) eventClrTypeName));
                if (!(@event is BaseEvent))
                {
                    throw new MessageDeserializationException((string) eventClrTypeName, metadataString);
                }

                return @event as T;
            }

            private static IEventStoreConnection GetEventStoreConnection()
            {
                var connectionString = System.Configuration.ConfigurationManager.ConnectionStrings["EventStore"].ConnectionString;
                var connection = EventStoreConnection.Create(connectionString);
                connection.ConnectAsync().Wait();
                return connection;
            }

            private static string GetStreamName<T>() where T : BaseEvent
            {
                return "$et-" + typeof(T).Name;
            }

要读取事件,您可以使用此代码段

 StreamEventsSlice currentSlice;
            long nextSliceStart = StreamPosition.Start;
            const int sliceCount = 200;

            do
            {
                currentSlice = await esConnection.ReadStreamEventsForwardAsync(streamName, nextSliceStart, sliceCount, true);

                foreach (var @event in currentSlice.Events)
                {
                    var myEvent = GetInstanceOfEvent<OrderMerchantFeesCalculatedEvent>(@event);


                    TransformEvent(myEvent);
                }

                nextSliceStart = currentSlice.NextEventNumber;

            } while (currentSlice.IsEndOfStream == false);