cqrs-rust-lib
A pragmatic CQRS / Event Sourcing library for Rust with pluggable storage backends, structured domain errors, and REST integration.
Features
- Split
Aggregate/CommandHandlertraits (Single Responsibility) - Structured domain errors —
CqrsError+define_domain_errors!macro - Pluggable storage backends: InMemory, MongoDB, PostgreSQL, SurrealDB
- Unified
Querytrait — auto-derives filter from struct fields (RSQL under the hood) - HTTP Codex convention —
CqrsHttpQuery<Q>extracts_q,page,page_size,sortfrom HTTP params - Backend prelude pattern — swap the entire backend with one
useline - REST routers with Axum and auto-generated OpenAPI/Swagger (feature:
rest) - Audit log router for event history
- Snapshot support
- WASM-compatible core (no Tokio in production deps)
Installation
[]
= { = "0.7", = ["postgres"] }
Feature flags
| Feature | Description |
|---|---|
mongodb |
MongoDB event store + read storage |
postgres |
PostgreSQL event store + read storage |
surrealdb |
SurrealDB event store + read storage |
utoipa |
OpenAPI schema derives only (WASM-compatible) |
rest |
Axum routers + OpenAPI (implies utoipa, native only) |
all |
rest + mongodb + postgres + surrealdb |
Quick Start
1. Define your domain
use ;
use ;
2. Execute commands
use ;
use ;
let store = new;
let engine = new;
let ctx = default;
let id = engine.execute_create.await?;
engine.execute_update.await?;
Domain Error Codes
use ;
use StatusCode;
define_domain_errors!
Response shape:
Backend Preludes
Each backend exposes canonical type aliases under cqrs_rust_lib::prelude::<backend>.
Swapping the backend requires changing a single import line — the rest of the wiring is identical.
// Change only this line to swap backends:
use postgres as db;
// use cqrs_rust_lib::prelude::mongodb as db;
// use cqrs_rust_lib::prelude::surrealdb as db;
// Everything below stays the same:
let es = new;
let repo = new;
let snap = new;
| Alias | inmemory | postgres | mongodb | surrealdb |
|---|---|---|---|---|
EventStorePersist |
✓ | ✓ | ✓ | ✓ |
ReadStorage |
— | ✓ | ✓ | ✓ |
FromSnapshotStorage |
— | ✓ | ✓ | ✓ |
The connection setup (client, pool, URI) is necessarily backend-specific and stays outside the prelude.
Query Trait (Read Side)
Query is the unified read-side filter/pagination/sort interface. It requires Serialize (supertrait) so that equality filters are auto-derived from struct fields — no boilerplate needed in most cases.
use Query;
use ;
// Empty impl: filter auto-derived, no pagination override, no sort
Override only what you need:
use ;
use ;
HTTP Codex convention (feature: rest)
CqrsHttpQuery<Q> is an Axum extractor that adds _q (RSQL), page, page_size, sort on top of any typed Q. Use it directly with CQRSCodexReadRouter:
use ;
// Routes with HTTP Codex params: GET /games?_q=available==true&page=0&page_size=20&sort=-title
routes
Filter priority: _q (RSQL) AND Q::filter() — combined. Sort priority: HTTP sort → Q::sort() → Q::default_sort().
Storage Backends
PostgreSQL
use postgres as db;
use NoTls;
let = connect.await?;
spawn;
let client = new;
client.batch_execute.await?;
let es = from_client;
MongoDB
use mongodb as db;
let options = parse.await?;
let db_client = with_options?;
let database = db_client.database;
let es = new;
SurrealDB
use surrealdb as db;
use connect;
let surreal = connect.await?;
surreal.use_ns.use_db.await?;
surreal.query.await?.check?;
let es = new;
REST Routers (feature: rest)
use ;
// Standard router — typed query params only
routes
// Codex router — adds _q, page, page_size, sort HTTP params
routes
// Write + audit
routes
routes
See example/todolist/src/api.rs for complete wiring with Swagger UI.
Architecture
Aggregate (state + events) CommandHandler (commands → events)
\ /
CqrsCommandEngine ────── EventStore (persist)
│ │
Dispatchers Storage backends
(projections) (InMemory / PG / Mongo / Surreal)
│
ReadStorage ← Query (filter + sort + pagination)
Key Types
| Type | Description |
|---|---|
Aggregate |
Domain state, event application, identity |
CommandHandler |
Command processing, business validation |
CqrsCommandEngine |
Orchestrates command execution |
EventStore / EventStoreImpl |
Event persistence abstraction |
CqrsError |
Unified structured error type |
CqrsContext |
Carries user, request ID, correlation ID |
Dispatcher |
Reacts to persisted events (projections / views) |
View |
Read model projection |
Query |
Read-side filter / pagination / sort interface |
CqrsHttpQuery<Q> |
HTTP Codex extractor wrapping a typed Q |
Examples
| Example | Storage | Highlights |
|---|---|---|
example/bank |
MongoDB | Domain errors (prefix 10), views, movements sub-resource |
example/todolist |
PostgreSQL | REST API, Swagger UI, snapshots, integration tests |
example/ludotheque |
SurrealDB | Full pipeline: event store + view + filter + sort |
Migration Guides
License
MIT — see LICENSE.