Stream is a sequence of events made available over time ordered by revision.
Streams are used in many areas, here is just few examples of event streams:
- State of DDD aggregate.
- Financial transactions of user.
- Sequence of telemetry data from IoT device.
- etc.
An event is a fact. The domain may be updated to reflect the fact represented by the event.
Each event has a unique identifier, a timestamp, and a payload.
A stream identifier is a unique identifier of the stream.
It is an abstraction of the sequence of events, guaranteeing uniqueness and order, as well isolation
.
Example of stream IDs:
- ID of DDD aggregate.
- ID of user.
- ID of IoT device.
- etc.
Event identifier is a unique identifier of the event. Guarantee uniqueness of the event in the stream.
Revision is a sequence number of the event in the stream. It is used to:
- Order events in the stream.
- Establish optimistic concurrency control.
Despite the fact that usually event timestamp is used to order events, it is not really reliable, since there could be a clock skew between different part of the system in distributed architecture.
Stream store is an abstraction of particular storage that allows to read and write events to the stream sequentially.
-
Id
is a value object (immutable class) that has implicit conversion from and to string.Thus you don't need to create Id object explicitly and use
ToString()
to convert to string back.
Also implementsIEquatable
for itself and forString
. -
Revision
is a value object (immutable class) that represents a revision of the stream.
It is used for optimistic concurrency control and event ordering. It has implicit conversion from and toInt32
type.
Also implementsIEquatable
andIComparable
for itself and forInt32
. -
You can read from any stream starting from the provided revision.
-
ReadToEnd
method returns collection of events from the stream starting from the provided revision:- Contains only unique events ordered by revision.
- Contains only events that were committed.
-
Stream revision is always the revision of an event with maximum revision value.
-
Idempotency of reading and deletion fully depends on particular storage implementation.
-
You don't need to retrieve stream to add events to it.
Appending to stream and getting stream are separate operations. -
Despite the fact that reading is declared as asynchronous and iterative operation, for the sake of performance it is implemented as paginated operation.
You can define the page size by using
WithReadingPageSize
method of store configuration, by default it is 10 events. -
Reading and writing operations are not thread-safe.
Thus, it is not recommended to use the same instances ofIStreamWriter
orIAsyncEnumerable<StreamEvent>
in multiple threads.