EventDBX
You'll likely enjoy this database system. Worry less about how you structure your data and focus more on your business logic.
Overview
EventDBX is an event-sourced, key-value, write-side database system designed to provide immutable, append-only storage for events across various domains. It is ideal for applications requiring detailed audit trails for compliance, complex business processes involving states, and high data integrity levels.
Getting Started
Follow the steps below to spin up EventDBX locally. All commands are expected to run from the repository root unless stated otherwise.
-
Install prerequisites
- Rust toolchain (edition 2024) via
rustup rocksdbis vendored through the Rust crate, so no extra native packages are required.
- Rust toolchain (edition 2024) via
-
Clone and build
-
(Optional) Run checks
-
Create the initial configuration Supply a 32-byte data-encryption key (DEK) encoded as Base64. You can generate one on macOS/Linux with
openssl rand -base64 32. -
Start the server
- Omit
--foregroundto daemonise the process. - Use
--data-dir <path>to override the default./.eventdbxdirectory. - Restriction (schema enforcement) is enabled by default; disable it with
--restrict=falseif you need a permissive environment.
- Omit
- Choose the API surface with
--api rest,--api graphql,--api grpc, or--api all(enable every surface).--api grpc/--api allautomatically flip the gRPC listener on for the current session; persistently enable it by settinggrpc.enabled = trueinconfig.toml.
-
Define a schema (recommended when running in restricted mode)
-
Issue a token for CLI access
-
Append an event
You now have a working EventDBX instance with an initial aggregate. Explore the Command-Line Reference for the full set of supported operations.
Features
- Flexible JSON payloads: Events accept arbitrary JSON payloads; scalar values are normalized into strings for state tracking, while structured objects remain fully queryable.
- Immutable Data Structure: Once data is entered into EventDBX, it becomes immutable, meaning it cannot be altered or deleted. This characteristic is crucial for applications where the accuracy and traceability of historical data are paramount, such as medical records, financial transactions, and supply chain management. Data can be archived, moving from short-term to long-term storage, but cannot be deleted.
- Event Sourcing and Replay: EventDBX is built on the principle of event sourcing, storing all changes to the data as a sequence of events. This allows for the complete replay of events to reconstruct the database's state at any point in time, thereby enhancing data recovery and audit capabilities. Unlike traditional databases that execute update statements to modify data, this system is event-driven. Aggregate state changes are defined in the event object, allowing these events to be replayed at any time to reconstruct the aggregate's current state.
- Merkle Tree Integration: Each aggregate in EventDBX is associated with a Merkle tree of events, enabling verification of data integrity. The Merkle tree structure ensures that any data tampering can be detected, offering an additional security layer against data corruption.
- Built-in Audit Trails: EventDBX automatically maintains a comprehensive audit trail of all transactions, a feature invaluable for meeting compliance and regulatory requirements. It provides transparent and tamper-evident records. During audits, administrators can issue specific tokens to auditors to access and review specific aggregate instances and all relevant events associated with those instances.
- Security with Token-Based Authorization: EventDBX implements token-based authorization to manage database access. This approach allows for precise control over who can access and modify data, protecting against unauthorized changes. Unlike systems where a single application user account performs CRUD operations, EventDBX mandates that each user of the application obtains their own access token with a specific time horizon. For example, a doctor will receive their own access token, linked to their identifier (IAM-managed outside EventDBX), generated by the system for each event they handle, rather than having a single application user manage all CRUD operations. This ensures the system can accurately track that it was the doctor who made changes to an aggregate state, not the application.
- Encrypted Payloads & Secrets at Rest: Event payloads, aggregate snapshots, and
tokens.jsonare encrypted transparently when a DEK is configured. Metadata such as aggregate identifiers, versions, and Merkle roots remain readable so plugins, replication, and integrity checks keep working without additional configuration. - Powered by RocksDB and Rust: At its core, EventDBX utilizes RocksDB for storage, taking advantage of its high performance and efficiency. The system is developed in Rust, known for its safety, efficiency, and concurrency capabilities, ensuring that it is both rapid and dependable.
Restriction Modes
EventDBX can run in two validation modes, tuned for different phases of development:
- Unrestricted mode (
--restrict=false): Ideal for prototyping and rapid application development. Event payloads are accepted as-is with no schema definition or data-type friction, letting you iterate quickly without pre-registering aggregates, tables, or column types. - Restricted mode (default): Enables schema enforcement. Once your application matures, define required fields, data types, and validation rules via
eventdbx schema …commands. Incoming events must match the declared schema, and violations are rejected before they reach storage. You can toggle restriction per server start, so development and production can adopt different policies.
Command-Line Reference
EventDBX ships a single eventdbx binary. Every command accepts an optional --config <path> to point at an alternate configuration file.
Server lifecycle
eventdbx start [--port <u16>] [--data-dir <path>] [--foreground] [--restrict | --restrict=false]
Launches the server. Schema validation is enforced by default; pass--restrict=falseto run in permissive mode.eventdbx stop
Stops the running daemon referenced by the PID file.eventdbx status
Prints the current port, PID, uptime, and whether restriction is enabled.eventdbx restart [start options…]
Stops the existing daemon (if any) and restarts it with the provided options.eventdbx destroy [--yes]
Removes the PID file, data directory, and configuration file after confirmation (or immediately with--yes).
Configuration
eventdbx config [--port <u16>] [--data-dir <path>] [--master-key <secret>] [--dek <secret>] [--memory-threshold <usize>] [--list-page-size <usize>] [--page-limit <usize>] [--plugin-max-attempts <u32>]
Persists configuration updates. The first invocation must include both--master-keyand--dek.--list-page-sizesets the default page size for aggregate listings (default 10),--page-limitcaps any requested page size across list and event endpoints (default 1000, alias--event-page-limit), and--plugin-max-attemptscontrols how many retries are attempted before an event is marked dead (default 10).
Tokens
eventdbx token generate --group <name> --user <name> [--expiration <secs>] [--limit <writes>] [--keep-alive]
Issues a new token tied to a Unix-style group and user.eventdbx token list
Lists all tokens with status, expiry, and remaining writes.eventdbx token revoke --token <value>
Revokes a token immediately.eventdbx token refresh --token <value> [--expiration <secs>] [--limit <writes>]
Extends the lifetime or write allowance of an existing token.
Schemas
eventdbx schema create --aggregate <name> --events <event1,event2,...> [--snapshot-threshold <u64>]eventdbx schema add --aggregate <name> --events <event1,event2,...>eventdbx schema remove --aggregate <name> --event <name>eventdbx schema list
Schemas are stored on disk; when the server runs with restriction enabled, incoming events must satisfy the recorded schema.
Aggregates
eventdbx aggregate apply --aggregate <type> --aggregate-id <id> --event <name> --field KEY=VALUE... [--stage]
Appends an event immediately—use--stageto queue it for a later commit.eventdbx aggregate list [--skip <n>] [--take <n>] [--stage]
Lists aggregates with version, Merkle root, and archive status; pass--stageto display queued events instead.eventdbx aggregate get --aggregate <type> --aggregate-id <id> [--version <u64>] [--include-events]eventdbx aggregate replay --aggregate <type> --aggregate-id <id> [--skip <n>] [--take <n>]eventdbx aggregate verify --aggregate <type> --aggregate-id <id>eventdbx aggregate snapshot --aggregate <type> --aggregate-id <id> [--comment <text>]eventdbx aggregate archive --aggregate <type> --aggregate-id <id> [--comment <text>]eventdbx aggregate restore --aggregate <type> --aggregate-id <id> [--comment <text>]eventdbx aggregate remove --aggregate <type> --aggregate-id <id>Removes an aggregate that has no events (version still 0).eventdbx aggregate commit
Flushes all staged events in a single atomic transaction.
Staged events are stored in .eventdbx/staged_events.json. Use aggregate apply --stage to add entries to this queue, inspect them with aggregate list --stage, and persist the entire batch with aggregate commit. Events are validated against the active schema (when restriction is enabled) during both staging and commit. The commit operation writes every pending event in one RocksDB batch, guaranteeing all-or-nothing persistence.
Plugins
eventdbx plugin map --aggregate <name> --field <field> --datatype <type>
Records the base column type for a field; add--plugin postgres [--plugin-name <label>]to override the Postgres mapping only.eventdbx plugin config postgres --connection <connection-string> [--name <label>] [--disable]eventdbx plugin config csv --name <label> --output-dir <dir> [--disable]eventdbx plugin config tcp --name <label> --host <hostname> --port <u16> [--disable]eventdbx plugin config http --name <label> --endpoint <host|url> [--https] [--header KEY=VALUE]... [--disable]eventdbx plugin config grpc --name <label> --endpoint <host|url> [--disable]eventdbx plugin config json --name <label> --path <file> [--pretty] [--disable]eventdbx plugin config log --name <label> --level <trace|debug|info|warn|error> [--template "text with {aggregate} {event} {id}"] [--disable]eventdbx plugin enable <label>eventdbx plugin disable <label>eventdbx plugin remove <label>eventdbx plugin testeventdbx plugin listeventdbx plugin queueeventdbx plugin queue cleareventdbx plugin queue retry [--event-id <uuid>]eventdbx plugin replay <plugin-name> <aggregate> [<aggregate_id>]
Clearing dead entries prompts for confirmation to avoid accidental removal. Manual retries run the failed events immediately; use --event-id to target a specific entry.
Replication
eventdbx remote add <name> <endpoint> --public-key <base64>
Registers a standby and pins its Ed25519 public key.eventdbx remote rm <name>
Removes a configured remote.eventdbx remote ls
Lists remotes with their endpoints.eventdbx remote show <name>
Displays the endpoint and pinned key for a remote.eventdbx remote key [--show-path]
Prints this node's replication public key (generated on first run).eventdbx remote push <name> [--dry-run] [--batch-size <n>] [--aggregate <type>...] [--aggregate-id <type:id>...]
Streams local events to the remote in fast-forward mode; dry runs report pending changes.eventdbx remote pull <name> [--dry-run] [--batch-size <n>] [--aggregate <type>...] [--aggregate-id <type:id>...]
Fast-forwards the local node from the remote, reporting changes in dry-run mode.
Replication keys live alongside the data directory (replication.key / replication.pub) and are created automatically the first time the CLI loads configuration. The standby gRPC listener defaults to 127.0.0.1:7443; override it in config.toml via replication.bind_addr when you expose the replica on another interface. When the HTTP server processes writes it streams committed events to every configured remote over gRPC using the pinned public keys. Use --aggregate repeatedly to scope push/pull to specific aggregate types when you only need to sync a subset of data, and --aggregate-id TYPE:ID to target individual aggregates.
Use --aggregate repeatedly to scope push/pull to specific aggregate types when you only need to sync a subset of data.
Maintenance
eventdbx backup --output <path> [--force]
Creates a compressed archive with the entire EventDBX data directory and configuration. Stop the server before running a backup to avoid partial snapshots.eventdbx restore --input <path> [--data-dir <path>] [--force]
Restores data from a backup archive. Use--data-dirto override the stored location, and--forceto overwrite non-empty destinations. The server must be stopped before restoring.
Plugins fire after every committed event to keep external systems in sync. Each plugin sends or records different data:
Failed deliveries are automatically queued and retried with exponential backoff. The server keeps attempting until the plugin succeeds or the aggregate is removed, ensuring transient outages do not drop notifications.
Use eventdbx plugin queue to inspect pending/dead event IDs.
Plugin configurations are stored in .eventdbx/plugins.json. Each plugin instance requires a unique --name so you can update, enable, disable, remove, or replay it later. plugin enable validates connectivity (creating directories, touching files, or checking network access) before marking the plugin active. Remove a plugin only after disabling it with plugin disable <name>. plugin replay resends stored events for a single aggregate instance—or every instance of a type—through the selected plugin.
- Postgres: Upserts aggregate state into a Postgres table, expanding columns based on schema mappings or
plugin map --plugin postgresoverrides. - CSV: Appends state snapshots into
<aggregate>.csv, expanding columns as new fields appear. - TCP: Writes a single-line JSON
EventRecordto the configured socket. - HTTP: POSTs the
EventRecordJSON to the endpoint with optional headers; add--httpsduring configuration to force HTTPS when the endpoint lacks a scheme. - gRPC: Sends
EventRecordbatches to a remote gRPC endpoint compatible with the replicationApplyEventsAPI.
Example gRPC configuration:
# point at an existing replication-compatible listener
# enable the plugin once connectivity is confirmed
# inspect status
- JSON: Appends the
EventRecordJSON (pretty if requested) to the given file. - Log: Emits a formatted line via
tracingat the configured level. By default:aggregate=<type> id=<id> event=<event>.
Example TCP/HTTP/JSON payload (EventRecord):
REST API
The server exposes a small HTTP API (served on port 7070 by default). All endpoints require a bearer token unless otherwise noted.
| Method & Path | Description |
|---|---|
GET /health |
Liveness probe (unauthenticated). |
GET /v1/aggregates |
Lists aggregates; supports skip/take query parameters. |
GET /v1/aggregates/{aggregate_type}/{aggregate_id} |
Returns the current state for a specific aggregate. |
GET /v1/aggregates/{aggregate_type}/{aggregate_id}/events |
Lists events for an aggregate; supports skip/take pagination. |
POST /v1/events |
Appends an event; aggregate identifiers are provided in the body. |
GET /v1/aggregates/{aggregate_type}/{aggregate_id}/verify |
Computes and returns the Merkle root for integrity verification. |
Paginated responses cap take at the configurable page_limit (default 1000). Adjust it with eventdbx config --page-limit <n> if you need larger pages.
All authenticated requests must include Authorization: Bearer <token> with a token issued via the CLI.
cURL examples
# Post a JSON event (global endpoint)
# Retrieve the first 10 events for an aggregate (supports `skip`/`take` up to `page_limit`)
# Health check
# List aggregates (replace TOKEN with an active value)
# Append an event (global endpoint)
GraphQL API
GraphQL is served on /graphql, and an interactive Playground is available via GET /graphql or /graphql/playground. Supply the same bearer token header used for REST requests.
gRPC API
Set grpc.enabled = true in config.toml to expose a gRPC surface (disabled by default). The listener binds to grpc.bind_addr (default 127.0.0.1:7442). The service mirrors the REST operations (AppendEvent, ListAggregates, GetAggregate, ListEvents, and VerifyAggregate) plus a simple Health probe.
Example grpcurl invocation:
Query example
Sample response:
Mutation example
The mutation triggers the same validation, replication, and plugin notifications as the REST endpoint.
My Unpopular Opinions
I believe databases—like Oracle, PostgreSQL, and MongoDB—are bloated with unnecessary features. They bundle in query languages, stored procedures, search layers, and business logic that blur the line between data storage and application behavior.
A database’s role should be simple: store state. No SQL, no procedures, no functions, no search, no views—absolutely no business logic. A write-only database should be immutable: every change is an event with a purpose, not just an update.
EventDBX follows this principle. It focuses entirely on the write side of the CQRS design pattern, excelling at persisting state, tracking its evolution over time, and notifying downstream systems whenever a change occurs—so other services can react intelligently to what has changed. It leaves reading, searching, filtering, aggregation, and analytics to other systems.
— Patrick T.
Contributing
Quick Start
- Report Issues: Found a bug or have a suggestion? Open an issue with detailed information.
- Contribute Code:
- Fork & Clone: Fork the EventDBX repo and clone your fork.
- Branch: Create a branch for your changes from
develop. - Develop: Make your changes, adhering to our coding standards. Add or update tests as necessary.
- Commit: Use Conventional Commits for clear, structured commit messages (e.g.,
feat: add new featureorfix: correct a bug). - Pull Request: Submit a pull request (PR) against the
developbranch of the original repo. Describe your changes and link to any related issues.
Guidelines
- Formatting: A project-wide Prettier configuration lives at
.prettierrc.json; use it for Markdown/JSON/YAML changes. - Commit linting: Conventional Commit headers are enforced through
.commitlintrc.json. - Code Review: Your PR will be reviewed by the team. Be open to feedback and make requested adjustments.
- Merge: Once approved, your PR will be merged into the project, and you'll be credited as a contributor.
License
EventDBX is licensed under the MIT License.