Performance can become an issue when an aggregate has a large amout of historical events, because all these events must be loaded and compiled to calculate the current state of the aggregate. Snapshot can help resolving this issue.
The idea is simple: You take a snapshot of the aggregate when necessary and save it in another store (snapshot store). Then to rehydrate the aggregate again you can save IO and computation by only retrieving the snapshot and events that occurred after that snapshot.
To enable this feature you need to implement the following:
-
Add the following constructor to the domain aggregate:
public GiftCard( Guid id, IAggregateSnapshot<Guid> snapshot, IEnumerable<IAggregateEvent<Guid>> savedEvents) : base(id, snapshot, savedEvents) { }
-
Create a class that represents the snapshot
public class GiftCardSnapshot : AggregateSnapshot<Guid> { public GiftCardSnapshot(Guid aggregateId, int aggregateVersion, decimal balance) : base(aggregateId, aggregateVersion) { Balance = balance; } public decimal Balance { get; } }
-
Override the
CreateSnapshot()
method in aggregateprotected override IAggregateSnapshot<Guid> CreateSnapshot() { return new GiftCardSnapshot(Id, Version, Balance); }
-
Override the
ApplySnapshot()
methodprotected override void ApplySnapshot(IAggregateSnapshot<Guid> snapshot) { GiftCardSnapshot giftCardSnapshot = snapshot as GiftCardSnapshot ?? throw new InvalidOperationException(); Balance = giftCardSnapshot.Balance; }
Note:
- It is your resposibility to ensure that your snapshot class contains all data needed to restore that aggregate to the corresponding version.
- To use built-in stores, make sure that your snapshot class can be serialized/deserialized by Json.NET
-
Register snapshot store in
IEventSourcingBuilder
, in a similar way as registering the event stores, e.g.,services.AddEventSourcing(builder => builder .UseTextFileEventStore<GiftCard, Guid>(x => x.Folder = "C:/Temp/GiftcardEvents") .UseTextFileSnapshotStore<GiftCard, Guid>(x => x.Folder = "C:/Temp/GiftcardEvents"));
Note: You can store snapshot in the same or in a different storage with events. For example: store events in DynamoDB and store snapshots in text files:
services.AddEventSourcing(builder => builder .UseDynamoDBEventStore<GiftCard, Guid>(x => x.TableName = "GiftcardEvents") .UseTextFileSnapshotStore<GiftCard, Guid>(x => x.Folder = "C:/Temp/GiftcardEvents"));
-
Optionally you can programatically initialize snapshot store in a similar way with event store (see here):
public void Configure(IApplicationBuilder app, IWebHostEnvironment env, IEventStoreInitializer<GiftCard, Guid> eventStoreInitializer, ISnapshotStoreInitializer<GiftCard, Guid> snapshotStoreInitializer) { eventStoreInitializer.EnsureCreatedAsync().Wait(); snapshotStoreInitializer.EnsureCreatedAsync().Wait(); // configure http request pipeline }
-
It's totally up to you to decide when to take an snapshot by calling
aggregate.TakeSnapshot()
. The snapshot will be saved when invokedAggregateRepository<TAggregate, TKey>.SaveAggregateAsync()
-
By default when calling
AggregateRepository<TAggregate, TKey>.FindAggregateAsync()
it automatically retrieve the last snapshot. You can force not loading any snapshot by calling:FindAggregateAsync(aggregateId, ignoreSnapshot: true)