Skip to main content

Rillo is a lightweight, type-safe Event Sourcing framework for Python, built on top of Pydantic.

Project description

Rillo

Rillo is a lightweight, type-safe Event Sourcing framework for Python, built on top of Pydantic.

Installation

Installing the core library using pip:

pip install rillo

Install with NATS JetStream support for repositories and snapshot stores:

pip install 'rillo[nats]'

Installing using uv:

uv add rillo
uv add rillo[nats]

Usage

Defining an Aggregate

Rillo uses Pydantic models for Events and State. Creating an Aggregate requires defining your State, Events, and mapping changes via the @mutator decorator.

from typing import Literal
from pydantic import BaseModel
from rillo import Aggregate, mutator

# 1. Define events
class UserSignedUp(BaseModel):
    type: Literal["UserSignedUpV1"] = "UserSignedUpV1"
    username: str

class AccountDeleted(BaseModel):
    type: Literal["AccountDeletedV1"] = "AccountDeletedV1"

# 2. Define aggregate state
class UserState(BaseModel):
    type: Literal["UserStateV1"] = "UserStateV1"
    username: str
    account_deleted: bool

# 3. Create the Aggregate
class User(Aggregate[UserState]):

    # Mutators specify how an event modifies the internal state
    @mutator
    def on_user_signed_up(self, event: UserSignedUp) -> None:
        self._state = UserState(
            username=event.username,
            account_deleted=False,
        )

    @mutator
    def on_account_deleted(self, event: AccountDeleted) -> None:
        if self._state is not None:
            self._state.account_deleted = True

    # Business logic generates and applies new events
    def sign_up(self, username: str) -> None:
        if self._state is not None:
            raise ValueError("User already exists.")

        self._apply(UserSignedUp(username=username))

    def delete_account(self) -> None:
        if self._state is None:
            raise ValueError("User does not exist.")

        self._apply(AccountDeleted())

# Using the aggregate
user = User(id="user-1")
user.sign_up("alice")
user.delete_account()

# Pending events are stored and ready to be committed
events = user.pending_events

Repositories and Snapshot Stores

Rillo provides a Repository base class to save/load events and a SnapshotStore base for capturing aggregate snapshots to optimize load times. Both have built-in support for NATS JetStream (NATSRepository & NATSSnapshotStore).

import asyncio
from nats.aio.client import Client as NATS
from rillo.nats import NATSRepository

async def main():
    nc = NATS()
    await nc.connect("nats://localhost:4222")
    js = nc.jetstream()

    # Create a repository instance
    repository = NATSRepository[User](
        js=js,
        stream_name="USERS",
        subject_prefix="users.events"
    )

    user = User("user-123")
    user.sign_up("alice")

    # Persist pending events into NATS JetStream
    await repository.save(user)

    # Rehydrate aggregate state back from the event stream
    loaded_user = User("user-123")
    await repository.load(loaded_user)

if __name__ == "__main__":
    asyncio.run(main())

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

rillo-0.1.0.tar.gz (4.9 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

rillo-0.1.0-py3-none-any.whl (6.7 kB view details)

Uploaded Python 3

File details

Details for the file rillo-0.1.0.tar.gz.

File metadata

  • Download URL: rillo-0.1.0.tar.gz
  • Upload date:
  • Size: 4.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.11.7 {"installer":{"name":"uv","version":"0.11.7","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for rillo-0.1.0.tar.gz
Algorithm Hash digest
SHA256 c270ad96ce22745d3c0f4a1a5dbd024c734e4f873c8ede3ecbc6634ea418b365
MD5 4981201c97eb099b46758e1815e1af6f
BLAKE2b-256 dc2043b30b3082984bac85c178d96f8c138f7866d60027f0581c3496ce9820a1

See more details on using hashes here.

File details

Details for the file rillo-0.1.0-py3-none-any.whl.

File metadata

  • Download URL: rillo-0.1.0-py3-none-any.whl
  • Upload date:
  • Size: 6.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.11.7 {"installer":{"name":"uv","version":"0.11.7","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for rillo-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 19737e427e50c5a844f6e42fb982a6026a9039f97e8b000607542a4d9ffd9f13
MD5 d7358f19af67a5179489b1cba356e716
BLAKE2b-256 f4d6d3465342541de3ae7b82a78b3d30ff34d87f8841e8fac381b66126b18e16

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page