Tutorial, practical samples and other resources about Event Sourcing in .NET Core.
- EventSourcing.NetCore
- 1. Event Sourcing
- 2. Support
- 3. Prerequisites
- 4. Libraries used
- 5. Articles
- 6. Event Store - Marten
- 7. Message Bus (for processing Commands, Queries, Events) - MediatR
- 8. CQRS (Command Query Responsibility Separation)
- 9. Fully working sample application
- 10. Self-paced training Kit
- 11. NuGet packages to help you get started.
- 12. Other resources
- 12.1 Introduction
- 12.2 Event Sourcing on production
- 12.3 Projections
- 12.4 Snapshots
- 12.5 Versioning
- 12.6 Storage
- 12.7 Design & Modeling
- 12.8 GDPR
- 12.9 Conflict Detection
- 12.10 Functional programming
- 12.12 Testing
- 12.13 CQRS
- 12.14 Tools
- 12.15 Event Sourcing vs Messaging
- 12.15 Event processing
- 12.16 Distributed processes
- 12.17 Domain Driven Design
- 12.18 Architecture Weekly
Event Sourcing is a design pattern in which results of business operations are stored as a series of events.
It is an alternative way to persist data. In contrast with state-oriented persistence that only keeps the latest version of the entity state, Event Sourcing stores each state change as a separate event.
Thanks for that, no business data is lost. Each operation results in the event stored in the databse. That enables extended auditing and diagnostics capabilities (both technically and business-wise). What's more, as events contains the business context, it allows wide business analysis and reporting.
In this repository I'm showing different aspects, patterns around Event Sourcing. From the basic to advanced practices.
Events, represent facts in the past. They carry information about something accomplished. It should be named in the past tense, e.g. "user added", "order confirmed". Events are not directed to a specific recipient - they're broadcasted information. It's like telling a story at a party. We hope that someone listens to us, but we may quickly realise that no one is paying attention.
Events:
- are immutable: "What has been seen, cannot be unseen".
- can be ignored but cannot be retracted (as you cannot change the past).
- can be interpreted differently. The basketball match result is a fact. Winning team fans will interpret it positively. Losing team fans - not so much.
Read more in my articles:
- π What's the difference between a command and an event?
- π Events should be as small as possible, right?
Events are logically grouped into streams. In Event Sourcing, streams are the representation of the entities. All the entity state mutations ends up as the persisted events. Entity state is retrieved by reading all the stream events and applying them one by one in the order of appearance.
A stream should have a unique identifier representing the specific object. Each event has its own unique position within a stream. This position is usually represented by a numeric, incremental value. This number can be used to define the order of the events while retrieving the state. It can be also used to detect concurrency issues.
Technically events are messages.
They may be represented, e.g. in JSON, Binary, XML format. Besides the data, they usually contain:
- id: unique event identifier.
- type: name of the event, e.g. "invoice issued".
- stream id: object id for which event was registered (e.g. invoice id).
- stream position (also named version, order of occurrence, etc.): the number used to decide the order of the event's occurrence for the specific object (stream).
- timestamp: representing a time at which the event happened.
- other metadata like
correlation id
,causation id
, etc.
Sample event JSON can look like:
{
"id": "e44f813c-1a2f-4747-aed5-086805c6450e",
"type": "invoice-issued",
"streamId": "INV/2021/11/01",
"streamPosition": 1,
"timestamp": "2021-11-01T00:05:32.000Z",
"data":
{
"issuer": {
"name": "Oscar the Grouch",
"address": "123 Sesame Street",
},
"amount": 34.12,
"number": "INV/2021/11/01",
"issuedAt": "2021-11-01T00:05:32.000Z"
},
"metadata":
{
"correlationId": "1fecc92e-3197-4191-b929-bd306e1110a4",
"causationId": "c3cf07e8-9f2f-4c2d-a8e9-f8a612b4a7f1"
}
}
Event Sourcing is not related to any type of storage implementation. As long as it fulfils the assumptions, it can be implemented having any backing database (relational, document, etc.). The state has to be represented by the append-only log of events. The events are stored in chronological order, and new events are appended to the previous event. Event Stores are the databases' category explicitly designed for such purpose.
Read more in my article:
Feel free to create an issue if you have any questions or request for more explanation or samples. I also take Pull Requests!
π If this repository helped you - I'd be more than happy if you join the group of my official supporters at:
π Github Sponsors
For running the Event Store examples you need to have:
- .NET 5 installed - https://dotnet.microsoft.com/download/dotnet/5.0
- Postgres DB. You can get it by:
- Installing Docker, going to the
docker
folder and running:
docker-compose up
- Installing a most recent version of the Postgres DB (eg. from https://www.postgresql.org/download/).
More information about using .NET Core, WebApi and Docker you can find in my other tutorials: WebApi with .NET
Watch "Practical Event Sourcing with Marten":
Slides:
- Practical Event Sourcing with Marten - EN, PL
- Ligths and Shades of Event-Driven Design - EN, PL
- Adventures in Event Sourcing and CQRS - PL
Read also more on the Event Sourcing and CQRS topics in my blog posts:
- π What's the difference between a command and an event?
- π Events should be as small as possible, right?
- π Why a bank account is not the best example of Event Sourcing?
- π How to (not) do the events versioning?
- π CQRS facts and myths explained
- π Can command return a value?
- π How to create projections of events for nested object structures?
- π What texting your Ex has to do with Event-Driven Design?
- π What if I told you that Relational Databases are in fact Event Stores?
- π Optimistic concurrency for pessimistic times
- π Outbox, Inbox patterns and delivery guarantees explained
- π Saga and Process Manager - distributed processes in practice
- Creating event store
- Event Stream - is a representation of the entity in event sourcing. It's a set of events that happened for the entity with the exact id. Stream id should be unique, can have different types but usually is a Guid.
- Stream starting - stream should be always started with a unique id. Marten provides three ways of starting the stream:
- calling StartStream method with a stream id
var streamId = Guid.NewGuid(); documentSession.Events.StartStream<IssuesList>(streamId);
- calling StartStream method with a set of events
var @event = new IssueCreated { IssueId = Guid.NewGuid(), Description = "Description" }; var streamId = documentSession.Events.StartStream<IssuesList>(@event);
- just appending events with a stream id
var @event = new IssueCreated { IssueId = Guid.NewGuid(), Description = "Description" }; var streamId = Guid.NewGuid(); documentSession.Events.Append(streamId, @event);
- calling StartStream method with a stream id
- Stream loading - all events that were placed on the event store should be possible to load them back. Marten allows to:
- get list of event by calling FetchStream method with a stream id
var eventsList = documentSession.Events.FetchStream(streamId);
- geting one event by its id
var @event = documentSession.Events.Load<IssueCreated>(eventId);
- get list of event by calling FetchStream method with a stream id
- Stream loading from exact state - all events that were placed on the event store should be possible to load them back. Marten allows to get stream from exact state by:
- timestamp (has to be in UTC)
var dateTime = new DateTime(2017, 1, 11); var events = documentSession.Events.FetchStream(streamId, timestamp: dateTime);
- version number
var versionNumber = 3; var events = documentSession.Events.FetchStream(streamId, version: versionNumber);
- timestamp (has to be in UTC)
- Stream starting - stream should be always started with a unique id. Marten provides three ways of starting the stream:
- Event stream aggregation - events that were stored can be aggregated to form the entity once again. During the aggregation, process events are taken by the stream id and then replied event by event (so eg. NewTaskAdded, DescriptionOfTaskChanged, TaskRemoved). At first, an empty entity instance is being created (by calling default constructor). Then events based on the order of appearance (timestamp) are being applied on the entity instance by calling proper Apply methods.
- Aggregation general rules
- Online Aggregation - online aggregation is a process when entity instance is being constructed on the fly from events. Events are taken from the database and then aggregation is being done. The biggest advantage of online aggregation is that it always gets the most recent business logic. So after the change, it's automatically reflected and it's not needed to do any migration or updates.
- Inline Aggregation (Snapshot) - inline aggregation happens when we take the snapshot of the entity from the DB. In that case, it's not needed to get all events. Marten stores the snapshot as a document. This is good for performance reasons because only one record is being materialized. The con of using inline aggregation is that after business logic has changed records need to be reaggregated.
- Reaggregation - one of the biggest advantages of the event sourcing is flexibility to business logic updates. It's not needed to perform complex migration. For online aggregation it's not needed to perform reaggregation - it's being made always automatically. The inline aggregation needs to be reaggregated. It can be done by performing online aggregation on all stream events and storing the result as a snapshot.
- reaggregation of inline snapshot with Marten
var onlineAggregation = documentSession.Events.AggregateStream<TEntity>(streamId); documentSession.Store<TEntity>(onlineAggregation); documentSession.SaveChanges();
- reaggregation of inline snapshot with Marten
- Event transformations
- Events projection
- Multitenancy per schema
- Initialization - MediatR uses services locator pattern to find a proper handler for the message type.
- Sending Messages - finds and uses the first registered handler for the message type. It could be used for queries (when we need to return values), commands (when we acting).
- No Handlers - when MediatR doesn't find proper handler it throws an exception.
- Single Handler - by implementing IRequestHandler we're deciding that this handler should be run asynchronously with other async handlers (so we don't wait for the previous handler to finish its work).
- More Than One Handler - when there is more than one handler registered MediatR takes only one ignoring others when Send method is being called.
- Publishing Messages - finds and uses all registered handlers for the message type. It's good for processing events.
- No Handlers - when MediatR doesn't find proper handler it throws an exception
- Single Handler - by implementing INotificationHandler we're deciding that this handler should be run asynchronously with other async handlers (so we don't wait for the previous handler to finish its work)
- More Than One Handler - when there is more than one handler registered MediatR takes all of them when calling Publish method
- Pipeline (to be defined)
See also fully working sample application in Sample Project
- See sample how Entity Framework and Marten can coexist together with CQRS and Event Sourcing
I prepared the self-paced training Kit for the Event Sourcing. See more in the Workshop description.
It's split into two parts:
Event Sourcing basics - it teaches the event store basics by showing how to build your Event Store on Relational Database. It starts with the tables setup, goes through appending events, aggregations, projections, snapshots, and finishes with the Marten
basics. See more in here.
- Streams Table
- Events Table
- Appending Events
- Optimistic Concurrency Handling
- Event Store Methods
- Stream Aggregation
- Time Travelling
- Aggregate and Repositories
- Snapshots
- Projections
- Projections With Marten
Event Sourcing advanced topics - it's a real-world sample of the microservices written in Event-Driven design. It explains the topics of modularity, eventual consistency. Shows practical usage of WebApi, Marten as Event Store, Kafka as Event bus and ElasticSearch as one of the read stores. See more in here.
- Meetings Management Module - the module responsible for creating, updating meeting details. Written in
Marten
in Event Sourcing pattern. Provides both write model (with Event Sourced aggregates) and read model with projections. - Meetings Search Module - responsible for searching and advanced filtering. Uses
ElasticSearch
as storage (because of its advanced searching capabilities). It's a read module that's listening for the events published by the Meetings Management Module.
I gathered and generalized all of the practices used in this tutorial/samples in Nuget Packages maintained by me GoldenEye Framework. See more in:
-
GoldenEye DDD package - it provides a set of base and bootstrap classes that helps you to reduce boilerplate code and help you focus on writing business code. You can find all classes like Commands/Queries/Event handlers and many more. To use it run:
dotnet add package GoldenEye.Backend.Core.DDD
-
GoldenEye Marten package - contains helpers, and abstractions to use Marten as document/event store. Gives you abstractions like repositories etc. To use it run:
dotnet add package GoldenEye.Backend.Core.Marten
The simplest way to start is installing the project template by running
dotnet new -i GoldenEye.WebApi.Template.SimpleDDD
and then creating a new project based on it:
dotnet new SimpleDDD -n NameOfYourProject
- π Greg Young - CQRS & Event Sourcing
- π Jay Kreps - Why local state is a fundamental primitive in stream processing
- π Microsoft - Exploring CQRS and Event Sourcing
- π° Lorenzo Nicora - A visual introduction to event sourcing and cqrs
- π Mathew McLoughlin - An Introduction to CQRS and Event Sourcing Patterns
- π Emily Stamey - Hey Boss, Event Sourcing Could Fix That!
- π Derek Comartin - Event Sourcing Example & Explained in plain English
- π Duncan Jones - Introduction to event sourcing and CQRS
- π Roman Sachse - Event Sourcing - Do it yourself series
- π Martin Fowler - The Many Meanings of Event-Driven Architecture
- π Martin Fowler - Event Sourcing
- π Dennis Doomen - 16 design guidelines for successful Event Sourcing
- π Martin Kleppmann - Event Sourcing and Stream Processing at Scale
- π Dennis Doomen - The Good, The Bad and the Ugly of Event Sourcing
- π Alexey Zimarev - DDD, Event Sourcing and Actors
- π Thomas BΓΈgh Fangel - Event Sourcing: Traceability, Consistency, Correctness
- π Joseph Choe - Event Sourcing, Part 1: User Registration
- π Steven Van Beelen - Intro to Event-Driven Microservices using DDD, CQRS & Event sourcing
- π Yves Lorphelin - The Inevitable Event-Centric Book
- π Leo Gorodinski - Scaling Event-Sourcing at Jet
- π EventStoreDB - Customers' case studies
- π P. Avery, R. Reta - Scaling Event Sourcing for Netflix Downloads
- π M. Overeem, M. Spoor, S. Jansen, S. Brinkkemper - An Empirical Characterization of Event Sourced Systems and Their Schema Evolution -- Lessons from Industry
- π Michiel Overeem - Event Sourcing after launch
- π Greg Young - A Decade of DDD, CQRS, Event Sourcing
- π M. Kadijk, J. Taal - The beautiful headache called event sourcing
- π Thomas Weiss - Planet-scale event sourcing with Azure Cosmos DB
- π D. Kuzenski, N. Piani - Beyond APIs: Re-architected System Integrations as Event Sourced
- π Greg Young - Why Event Sourced Systems Fail
- π Kacper Gunia - War Story: How a Large Corporation Used DDD to Replace a Loyalty System
- π Vladik Khononov - The Dark Side of Events
- π Pedro Costa - Migrating to Microservices and Event-Sourcing: the Dos and Dontβs
- π Dennis Doomen - An Event Sourcing Retrospective - The Good, The Bad and the Ugly
- π David Schmitz - Event Sourcing You are doing it wrong
- π Dennis Doomen - A recipe for gradually migrating from CRUD to Event Sourcing
- π Alexey Zimarev - Projections in Event Sourcing
- π Rinat Abdulin - Event Sourcing - Projections
- π Derek Comartin - Projections in Event Sourcing: Build ANY model you want!
- π Kacper Gunia - Event Sourcing: Snapshotting
- π Derek Comartin - Event Sourcing: Rehydrating Aggregates with Snapshots
- π Greg Young - Versioning in an Event Sourced System
- π Kacper Gunia - Event Sourcing: Snapshotting
- π M. Overeem, M. Spoor - The dark side of event sourcing: Managing data conversion
- π Savvas Kleanthous - Event immutability and dealing with change
- π Versioning in an Event Sourced System
- π Greg Young - Building an Event Storage
- π Andrii Litvinov - Event driven systems backed by MongoDB
- π Dave Remy - Turning the database inside out with Event Store
- π Mathias Verraes - DDD and Messaging Architectures
- π David Boike - Putting your events on a diet
- π Thomas Pierrain - As Time Goes Byβ¦ (a Bi-temporal Event Sourcing story)
- π Vaughn Vernon - Effective Aggregate Design Part I: Modeling a Single Aggregate
- π Derek Comartin - Aggregate (Root) Design: Separate Behavior & Data for Persistence
- π Mauro Servienti - All our aggregates are wrong
- π Microsoft - Domain events: design and implementation
- π Event Storming
- π Event Modeling
- π Wojciech SuwaΕa - Building Microservices On .NET Core β Part 5 Marten An Ideal Repository For Your Domain Aggregates
- π James Geall - Conflict Detection and Resolution in an EventSourced System
- π Lightbend - Data modelling for Replicated Event Sourcing
- π° Bartosz Sypytkowski - Collaborative Event Sourcing
- π Greg Young - CQRS
- π Jimmy Bogard - CQRS and REST: the perfect match
- π Mark Seemann - CQS versus server-generated IDs
- π Julie Lerman - Data Points - CQRS and EF Data Models
- π Marco BΓΌrckel - Some thoughts on using CQRS without Event Sourcing
- π Bertrand Meyer - Eiffel: a language for software engineering (CQRS introduced)
- π οΈ Marten - .NET Transactional Document DB and Event Store on PostgreSQL
- π οΈ EventStoreDB - The stream database built for Event Sourcing
- π οΈ GoldenEye - The CQRS flavoured framework that will speed up your WebAPI and Microservices development
- π οΈ Eventuous - Event Sourcing for .NET
- π οΈ SQLStreamStore - Stream Store library targeting RDBMS based implementations for .NET
- π οΈ Equinox - .NET Event Sourcing library with CosmosDB, EventStoreDB, SqlStreamStore and integration test backends
- π Kacper Gunia - EventStoreDB vs Kafka
- π Vijay Nair - Axon and Kafka - How does Axon compare to Apache Kafka?
- π Kamil Grzybek - The Outbox Pattern
- π Dev Mentors - Inbox & Outbox pattern - transactional message processing
- π Jeremy D. Miller - Jasper's "Outbox" Pattern Support
- π Gunnar Morling - Reliable Microservices Data Exchange With the Outbox Pattern
- π Microsoft - Asynchronous message-based communication
- π NServiceBus - Outbox
- π Alvaro Herrera - Implement SKIP LOCKED for row-level locks
- π Hector Garcaa-Molrna, Kenneth Salem - Sagas
- π Caitie McCaffrey - Applying the Saga Pattern
- π Udi Dahan - If (domain logic) then CQRS or Saga?
- π Gregor Hohpe - Starbucks Does Not Use Two-Phase Commit
- π Microsoft - Design Patterns - Saga distributed transactions pattern
- π Microsoft - Design Patterns - Choreography
- π Martin Schimak - Know the Flow! Events, Commands & Long-Running Services
- π Martin Schimak - Aggregates and Sagas are Processes
- π Chris Richardson - Using sagas to maintain data consistency in a microservice architecture
- π Thanh Le - What is SAGA Pattern and How important is it?
- π Jimmy Bogard - Life Beyond Distributed Transactions: An Apostate's Implementation - Relational Resources
- π Rinat Abdullin - Evolving Business Processes
- π Microsoft - A Saga on Sagas
- π NServiceBus - Sagas
- π NServiceBus sagas: Integrations
- π Denis Rosa (Couchbase) - Saga Pattern | Application Transactions Using Microservices
- π Eric Evans - DDD and Microservices: At Last, Some Boundaries!
- π Domain-Driven Design: The First 15 Years
- π Jimmy Bogard - Domain-Driven Design: The Good Parts
- π» Jakub Pilimon - DDD by Examples
- π DDD Quickly
- π Vaughn Vernon - Reactive DDD: Modeling Uncertainty
If you're interested in Architecture resources, check my other repository: https://github.com/oskardudycz/ArchitectureWeekly/.
It contains a weekly updated list of materials I found valuable and educational.
EventSourcing.NetCore is Copyright Β© 2017-2021 Oskar Dudycz and other contributors under the MIT license.