CQRS Starter Kit to start fast into DDD. Eventsourcing included. Some soldering required.
This module is based on the work of and started as a fork of snatalenko/node-cqrs, so most credits go to Mr. Natalenko.
Alpha-Warning: work in progress, not tested in production and subject of change in api, features and options
Goal of this module is to adopt the CQRS/DDD + ES in an "easy" and reusable manner to node applications. It should not provide a new framework or make strong assumptions on infrastructure. That being said, the initial iteration will target frameworks like fastify, express and alike to use this package within their ecosystem.
To get to a production ready setup it should provied a working example within a setup of
- storage and message adapters SHOULD NOT handle their connections, pools, reconnections within this modules
- storage and message adapters SHOULD reuse their clients as injected from the outside
- storage and message adapters SHOULD be injectable (basic reference implementations inlcluded)
- this module SHOULD NOT provide a framework,
- this module SHOULD NOT require a specific framework,
- this module SHOULD NOT assume any filestructure
- this module SHOULD NOT setup a domain on it's own
- this module SHOULD NOT depend on any given framework
As this is work in progress I'll keep the original README attached below:
The package provides building blocks for making a CQRS-ES application. It was inspired by Lokad.CQRS, but not tied to a specific storage implementation or infrastructure. It favors ES6 classes and dependency injection, so any components can be modified or replaced with your own implementations without hacks to the package codebase.
Documentation at node-cqrs.org
Your app is expected to operate with loosely typed commands and events that match the following interface:
declare interface IMessage {
type: string,
aggregateId?: string|number,
aggregateVersion?: number,
sagaId?: string|number,
sagaVersion?: number,
payload?: any,
context?: any
}
Domain business logic should be placed in Aggregate, Saga and Projection classes:
- Aggregates handle commands and emit events
- Sagas handle events and enqueue commands
- Projections listen to events and update views
Message delivery is being handled by the following services (in order of appearance):
- Command Bus delivers commands to command handlers
- Aggregate Command Handler restores an aggregate state, executes a command
- Event Store persists events and deliver them to event handlers (saga event handlers, projections or any other custom services)
- Saga Event Handler restores saga state and applies event
From a high level, this is how the command/event flow looks like:
You can find sample code of a User domain in the /examples folder.
Describe an aggregate that handles a command:
const { AbstractAggregate } = require('node-cqrs');
class UserAggregate extends AbstractAggregate {
static get handles() {
return ['createUser'];
}
createUser(commandPayload) {
// ...
}
}
Then register aggregate in the DI container. All the wiring can be done manually, without a DI container (you can find it in samples), but with container it’s just easier:
const { Container, InMemoryEventStorage } = require('node-cqrs');
const container = new Container();
container.register(InMemoryEventStorage, 'storage');
container.registerAggregate(UserAggregate);
container.createUnexposedInstances();
Then send a command:
const userAggregateId = undefined;
const payload = {
username: 'john',
password: 'test'
};
container.commandBus.send('createUser', userAggregateId, { payload });
Behind the scene, an AggregateCommandHandler will catch the command,
try to load an aggregate event stream and project it to aggregate state,
then it will pass the command payload to the createUser
handler we’ve defined earlier.
The createUser
implementation can look like this:
createUser(commandPayload) {
const { username, password } = commandPayload;
this.emit('userCreated', {
username,
passwordHash: md5Hash(password)
});
}
Once the above method is executed, the emitted userCreated event will be persisted and delivered to event handlers (sagas, projections or any other custom event receptors).
Now it’s time to work on a read model. We’ll need a projection that will handle our events. Projection must implement 2 methods: subscribe(eventStore)
and project(event)
.
To make it easier, you can extend an AbstractProjection
:
const { AbstractProjection } = require('node-cqrs');
class UsersProjection extends AbstractProjection {
static get handles() {
return ['userCreated'];
}
userCreated(event) {
// ...
}
}
By default, projection uses async InMemoryView
for inner view, but we’ll use Map
to make it more familiar:
class UsersProjection extends AbstractProjection {
get view() {
return this._view || (this._view = new Map());
}
// ...
}
With Map
view, our event handler can look this way:
class UsersProjection extends AbstractProjection {
// ...
userCreated(event) {
this.view.set(event.aggregateId, {
username: event.payload.username
});
}
}
Once the projection is ready, it can be registered in the DI container:
container.registerProjection(UsersProjection, 'users');
And accessed from anywhere in your app:
container.users
// Map { 1 => { username: 'John' } }
- editorconfig
- eslint
npm test -- --watch
- node >= 7.6
- visionmedia/debug (MIT License)