neos / eventstore
Store for Event-Sourced applications
Fund package maintenance!
Other
Installs: 37 813
Dependents: 6
Suggesters: 0
Security: 0
Stars: 0
Watchers: 7
Forks: 2
Open Issues: 2
Requires
- php: ^8.1
- ramsey/uuid: ^4.3
- webmozart/assert: ^1.10
Requires (Dev)
- phpstan/phpstan: ^1.10
- phpunit/phpunit: ^10
- roave/security-advisories: dev-latest
- squizlabs/php_codesniffer: ^4.0.x-dev
This package is auto-updated.
Last update: 2025-01-22 10:16:25 UTC
README
This package provides interfaces and helpers to create Event-Sourced systems with PHP
Scope
In contrast to Neos.EventSourcing this package provides merely the low-level building blocks, has just a couple of dependencies and is less opinionated.
Note
This package mostly contains interfaces and implementations of Data Transfer Objects. To actually persist events, a corresponding adapter package is required, for example neos/eventstore-doctrineadapter
Usage
Install via composer:
composer require neos/eventstore
Appending events
Commit a single event to the event store
$eventStore->commit( streamName: StreamName::fromString('some-stream'), events: new Event(EventId::create(), EventType::fromString('SomeEventType'), EventData::fromString('{"foo": "bar"}')), expectedVersion: ExpectedVersion::ANY(), );
Commit multiple events at once
$correlationId = Uuid::uuid4()->toString(); $eventStore->commit( streamName: StreamName::fromString('some-stream'), events: Events::fromArray([ new Event(EventId::create(), EventType::fromString('SomeEventType'), EventData::fromString('foo'), correlationId: $correlationId), new Event(EventId::create(), EventType::fromString('SomeOtherType'), EventData::fromString('bar'), correlationId: $correlationId])), ]), expectedVersion: ExpectedVersion::ANY(), );
Note Multiple events can only ever be appended to the same stream at once
Expected version
Event-sourced systems are eventual consistent. This basically means that the models, that are used to base decisions on, might be out of date.
The ExpectedVersion
can be used to make sure, that no new events where appended to the stream since the model was reconstituted from the events.
That mechanism can be used to implement event-sourced aggregates.
ExpectedVersion::ANY
ExpectedVersion::ANY()
(as used in the examples above) skips the version check entirely and should only be used if no hard constraints are required.
ExpectedVersion::NO_STREAM
ExpectedVersion::NO_STREAM()
can be used to make sure that a given stream does not yet exist.
This is useful for events that represent the creation of an entity:
$eventStore->commit( streamName: StreamName::fromString('customer-' . $customerId->value), events: new Event(EventId::create(), EventType::fromString('CustomerHasSignedUp'), EventData::fromString($customerData->toJson())), expectedVersion: ExpectedVersion::NO_STREAM(), );
If the same code was executed again (with the same $customId
) it would fail
ExpectedVersion::STREAM_EXISTS
ExpectedVersion::STREAM_EXISTS()
is basically the opposite of NO_STREAM
: It fails if no events have been committed to the stream before
Reading events
Iterate over all events of a stream
To load all events and output their sequence number and type:
foreach ($eventStore->load(StreamName::fromString('some-stream')) as $eventEnvelope) { echo $eventEnvelope->sequenceNumber->value . ': ' . $eventEnvelope->event->type->value . PHP_EOL; }
Filter event stream
EventStoreInterface::load()
expects a second, optional, argument that allows to filter the event stream.
This can be used to only load events of a certain type:
$stream = $eventStore->load( streamName: StreamName::fromString('some-stream'), filter: EventStreamFilter::create(eventTypes: EventTypes::create(EventType::fromString('SomeEventType'))) );
Note
Filtering events based on their type is a low-level optimization that is usually not required and should only be applied if needed
Navigate the event stream
The resulting EventStreamInterface
of the EventStoreInterface::load()
call is a lazy representation of the stream.
Depending on the actual implementation the events are only loaded whenever they are accessed.
The EventStreamInterface
provides four methods to affect ordering and window of the events to load:
withMinimumSequenceNumber()
to specify the lowestSequenceNumber
that should be included in the streamwithMaximumSequenceNumber()
to specify the highestSequenceNumber
that should be included in the streamlimit()
to specify the maximum number of events to load in totalbackwards()
to load events in descending order.
Usually the withMinimumSequenceNumber()
is used to only load events that have not been processed yet by an event handler,
but sometimes it can be useful to allow for arbitrary event stream navigation.
The following example will read at most 10 events with sequence number between 500 and 1000 in descending order:
$stream = $eventStore->load(StreamName::fromString('some-stream')) ->withMinimumSequenceNumber(SequenceNumber::fromInteger(500)) ->withMaximumSequenceNumber(SequenceNumber::fromInteger(1000)) ->limit(10) ->backwards();
Deleting events
Events form the unique source of truth in a system and thus should never be deleted. In theory. In practice, it can be useful to be able to remove streams that are not in use any longer.
The following example deletes all events from "some-stream":
$eventStore->deleteStream(StreamName::fromString('some-stream'));
Warning
For obvious reasons, this method should only be used with great care. Mostly there are better ways to solve issues that seem to require deletion of events. Also note, that some Event store implementations might not support this feature
More examples
@see tests for more examples.
Contribution
Contributions in the form of issues, pull requests or discussions are highly appreciated
License
See LICENSE