Skip to main content

Python package for eventsourcing with KurrentDB

Project description

Please note: following the rebranding of EventStoreDB to KurrentDB, this package is the rebranding of eventsourcing-eventstoredb. Please migrate your code to use the eventsourcing-kurrentdb package when you are ready.

Event Sourcing in Python with KurrentDB

This is an extension package for the Python eventsourcing library that provides a persistence module for KurrentDB and EventStoreDB. It uses the kurrentdbclient package to communicate with KurrentDB via the gRPC interface. It is tested with KurrentDB 25.0 and three previous LTS versions of EventStoreDB (24.10, 23.10, and 22.10) across Python versions 3.9 to 3.13.

Installation

Use pip to install the stable distribution from the Python Package Index.

$ pip install eventsourcing-kurrentdb

Please note, it is recommended to install Python packages into a Python virtual environment.

Getting started

Define aggregates and applications in the usual way. Please note, "streams" in KurrentDB are constrained to start from position 0, and this package expects the originator_version of the first event in an aggregate sequence to be 0, so you must set INITIAL_VERSION on your aggregate classes to 0.

from __future__ import annotations

from dataclasses import dataclass
from typing import TypedDict
from uuid import NAMESPACE_URL, UUID, uuid5

from eventsourcing.application import Application
from eventsourcing.domain import Aggregate, event


class TrainingSchool(Application[UUID]):
    def register(self, name: str) -> int:
        dog = Dog(name)
        recordings = self.save(dog)
        return recordings[-1].notification.id

    def add_trick(self, name: str, trick: str) -> int:
        dog = self._get_dog(name)
        dog.add_trick(trick)
        recordings = self.save(dog)
        return recordings[-1].notification.id

    def get_dog_details(self, name: str) -> DogDetails:
        dog = self._get_dog(name)
        return {"name": dog.name, "tricks": tuple(dog.tricks)}

    def _get_dog(self, name: str) -> Dog:
        return self.repository.get(Dog.create_id(name))


class Dog(Aggregate):
    INITIAL_VERSION = 0  # for KurrentDB

    @staticmethod
    def create_id(name: str) -> UUID:
        return uuid5(NAMESPACE_URL, f"/dogs/{name}")

    @dataclass(frozen=True)
    class Registered(Aggregate.Created):
        name: str

    @dataclass(frozen=True)
    class TrickAdded(Aggregate.Event):
        trick: str

    @event(Registered)
    def __init__(self, name: str) -> None:
        self.name = name
        self.tricks: list[str] = []

    @event(TrickAdded)
    def add_trick(self, trick: str) -> None:
        self.tricks.append(trick)


class DogDetails(TypedDict):
    name: str
    tricks: tuple[str, ...]

String IDs

If you want to use Python strings as your aggregate IDs, then please read this example in the Python eventsourcing library docs.

Configuring the application to use KurrentDB

Configure the TrainingSchool application to use KurrentDB with environment variables. You can configure an application with environment variables by setting them in the operating system environment, or by using the application constructor argument env, or by setting the application class attribute env.

Set PERSISTENCE_MODULE to 'eventsourcing_kurrentdb'. Also set KURRENTDB_URI to a KurrentDB connection string URI. This value will be used as the uri argument when the KurrentDBClient class is constructed by this package.

import os

os.environ["TRAININGSCHOOL_PERSISTENCE_MODULE"] = "eventsourcing_kurrentdb"
os.environ["KURRENTDB_URI"] = "esdb://localhost:2113?Tls=false"

If you are connecting to a "secure" KurrentDB server, and if the root certificate of the certificate authority used to generate the server's certificate is not installed locally, then also set environment variable KURRENTDB_ROOT_CERTIFICATES to an SSL/TLS certificate suitable for making a secure gRPC connection to the KurrentDB server(s). This value will be used as the root_certificates argument when the KurrentDBClient class is constructed by this package.

os.environ["KURRENTDB_ROOT_CERTIFICATES"] = "<PEM encoded SSL/TLS root certificates>"

Please refer to the kurrentdbclient documentation for details about starting a "secure" or "insecure" KurrentDB server, and the "kdb" and "kdb+discover" KurrentDB connection string URI schemes, and how to obtain a suitable SSL/TLS certificate for use in the client when connecting to a "secure" KurrentDB server.

After configuring environment variables, construct the application.

training_school = TrainingSchool()

Call application methods from tests and user interfaces.

training_school.register("Fido")
training_school.add_trick("Fido", "roll over")
training_school.add_trick("Fido", "play dead")
dog_details = training_school.get_dog_details("Fido")
assert dog_details["name"] == "Fido"
assert dog_details["tricks"] == ("roll over", "play dead")

To see the events have been saved in KurrentDB, we can reconstruct the application and get Fido's details again.

training_school = TrainingSchool()

dog_details = training_school.get_dog_details("Fido")

assert dog_details["name"] == "Fido"
assert dog_details["tricks"] == ("roll over", "play dead")

Eventually-consistent materialised views

To project the state of an event-sourced application "write model" into a materialised view "read model", first define an interface for the materialised view using the TrackingRecorder class from the eventsourcing library.

The example below defines methods to count dogs and tricks for the TrainingSchool application

from abc import abstractmethod

from eventsourcing.persistence import Tracking, TrackingRecorder


class MaterialisedViewInterface(TrackingRecorder):
    @abstractmethod
    def incr_dog_counter(self, tracking: Tracking) -> None:
        pass

    @abstractmethod
    def incr_trick_counter(self, tracking: Tracking) -> None:
        pass

    @abstractmethod
    def get_dog_counter(self) -> int:
        pass

    @abstractmethod
    def get_trick_counter(self) -> int:
        pass

The MaterialisedViewInterface can be implemented as a concrete view class using a durable database such as PostgreSQL.

The example below counts dogs and tricks in memory, using "plain old Python objects".

from eventsourcing.popo import POPOTrackingRecorder


class InMemoryMaterialiseView(POPOTrackingRecorder, MaterialisedViewInterface):
    def __init__(self) -> None:
        super().__init__()
        self._dog_counter = 0
        self._trick_counter = 0

    def incr_dog_counter(self, tracking: Tracking) -> None:
        with self._database_lock:
            self._assert_tracking_uniqueness(tracking)
            self._insert_tracking(tracking)
            self._dog_counter += 1

    def incr_trick_counter(self, tracking: Tracking) -> None:
        with self._database_lock:
            self._assert_tracking_uniqueness(tracking)
            self._insert_tracking(tracking)
            self._trick_counter += 1

    def get_dog_counter(self) -> int:
        return self._dog_counter

    def get_trick_counter(self) -> int:
        return self._trick_counter

Define how events will be processed using the Projection class from the eventsourcing library.

The example below processes Dog events. The Dog.Registered events are processed by calling incr_dog_counter() on the materialised view. The Dog.TrickAdded events are processed by calling incr_trick_counter().

from typing import Any

from eventsourcing.dispatch import singledispatchmethod
from eventsourcing.projection import Projection
from eventsourcing.utils import get_topic


class CountProjection(Projection[MaterialisedViewInterface]):
    topics = (
        get_topic(Dog.Registered),
        get_topic(Dog.TrickAdded),
    )

    @singledispatchmethod
    def process_event(self, event: Any, tracking: Tracking) -> None:
        pass

    @process_event.register
    def dog_registered(self, _: Dog.Registered, tracking: Tracking) -> None:
        self.view.incr_dog_counter(tracking)

    @process_event.register
    def trick_added(self, _: Dog.TrickAdded, tracking: Tracking) -> None:
        self.view.incr_trick_counter(tracking)

Run the projection with the ProjectionRunner class from the eventsourcing library.

The example below shows that when the projection is run, the materialised view is updated by processing the event of the upstream event-sourced TrainingSchool application. It also shows that when tricks are subsequently added to the application's aggregates, events continue to be processed, such that the trick counter is incremented in the downstream materialised view "read model".

import os
from eventsourcing.projection import ProjectionRunner

with ProjectionRunner(
    application_class=TrainingSchool,
    projection_class=CountProjection,
    view_class=InMemoryMaterialiseView,
) as runner:

    # Get "read model" instance from runner, because
    # state of materialised view is stored in memory.
    materialised_view = runner.projection.view

    # Wait for the existing events to be processed.
    materialised_view.wait(
        application_name=training_school.name,
        notification_id=training_school.recorder.max_notification_id(),
    )

    # Query the "read model".
    dog_count = materialised_view.get_dog_counter()
    trick_count = materialised_view.get_trick_counter()

    # Record another event in "write model".
    notification_id = training_school.add_trick("Fido", "sit and stay")

    # Wait for the new event to be processed.
    materialised_view.wait(
        application_name=training_school.name,
        notification_id=notification_id,
    )

    # Expect one trick more, same number of dogs.
    assert dog_count == materialised_view.get_dog_counter()
    assert trick_count + 1 == materialised_view.get_trick_counter()

    # Write another event.
    notification_id = training_school.add_trick("Fido", "jump hoop")

    # Wait for the new event to be processed.
    materialised_view.wait(
        training_school.name,
        notification_id,
    )

    # Expect two tricks more, same number of dogs.
    assert dog_count == materialised_view.get_dog_counter()
    assert trick_count + 2 == materialised_view.get_trick_counter()

See the Python eventsourcing package documentation for more information about projecting the state of an event-sourced application into materialised views that use a durable database such as SQLite and PostgreSQL.

More information

For more information, please refer to the Python eventsourcing library, the Python kurrentdbclient package, and the KurrentDB website.

Contributors

Install Poetry

The first thing is to check you have Poetry installed.

$ poetry --version

If you don't, then please install Poetry.

It will help to make sure Poetry's bin directory is in your PATH environment variable.

But in any case, make sure you know the path to the poetry executable. The Poetry installer tells you where it has been installed, and how to configure your shell.

Please refer to the Poetry docs for guidance on using Poetry.

Setup for PyCharm users

You can easily obtain the project files using PyCharm (menu "Git > Clone..."). PyCharm will then usually prompt you to open the project.

Open the project in a new window. PyCharm will then usually prompt you to create a new virtual environment.

Create a new Poetry virtual environment for the project. If PyCharm doesn't already know where your poetry executable is, then set the path to your poetry executable in the "New Poetry Environment" form input field labelled "Poetry executable". In the "New Poetry Environment" form, you will also have the opportunity to select which Python executable will be used by the virtual environment.

PyCharm will then create a new Poetry virtual environment for your project, using a particular version of Python, and also install into this virtual environment the project's package dependencies according to the pyproject.toml file, or the poetry.lock file if that exists in the project files.

You can add different Poetry environments for different Python versions, and switch between them using the "Python Interpreter" settings of PyCharm. If you want to use a version of Python that isn't installed, either use your favourite package manager, or install Python by downloading an installer for recent versions of Python directly from the Python website.

Once project dependencies have been installed, you should be able to run tests from within PyCharm (right-click on the tests folder and select the 'Run' option).

You should also be able to open a terminal window in PyCharm, and run the project's Makefile commands from the command line (see below).

Setup from command line

Obtain the project files, using Git or suitable alternative.

In a terminal application, change your current working directory to the root folder of the project files. There should be a Makefile in this folder.

Use the Makefile to create a new Poetry virtual environment for the project and install the project's package dependencies into it, using the following command.

$ make install

Please note, if you create the virtual environment in this way, and then try to open the project in PyCharm and configure the project to use this virtual environment as an "Existing Poetry Environment", PyCharm sometimes has some issues (don't know why) which might be problematic. If you encounter such issues, you can resolve these issues by deleting the virtual environment and creating the Poetry virtual environment using PyCharm (see above).

Project Makefile commands

You can start KurrentDB using the following command.

$ make start-kurrentdb

You can run tests using the following command (needs KurrentDB to be running).

$ make test

You can stop KurrentDB using the following command.

$ make stop-kurrentdb

You can check the formatting of the code using the following command.

$ make lint

You can reformat the code using the following command.

$ make fmt

Tests belong in ./tests. Code-under-test belongs in ./eventsourcing_kurrentdb.

Edit package dependencies in pyproject.toml. Update poetry.lock and installed packages using the following command.

$ make update

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

eventsourcing_kurrentdb-1.2.2.tar.gz (15.5 kB view details)

Uploaded Source

Built Distribution

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

eventsourcing_kurrentdb-1.2.2-py3-none-any.whl (11.3 kB view details)

Uploaded Python 3

File details

Details for the file eventsourcing_kurrentdb-1.2.2.tar.gz.

File metadata

  • Download URL: eventsourcing_kurrentdb-1.2.2.tar.gz
  • Upload date:
  • Size: 15.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.1.2 CPython/3.13.3 Darwin/24.4.0

File hashes

Hashes for eventsourcing_kurrentdb-1.2.2.tar.gz
Algorithm Hash digest
SHA256 6914767eede0ecb13f4c6993dc4f09ecf537b742b1c7c978b72381d595d22b43
MD5 002a6a4ade41e00dba78f86d46139167
BLAKE2b-256 c0b86a2ea5e81485641a95b70104a1aaf5bae68c9a9935241f71d207ba3114a6

See more details on using hashes here.

File details

Details for the file eventsourcing_kurrentdb-1.2.2-py3-none-any.whl.

File metadata

File hashes

Hashes for eventsourcing_kurrentdb-1.2.2-py3-none-any.whl
Algorithm Hash digest
SHA256 62a77219e25b64f782f9d55470700e7005028f97493d72641a98fbfacfc4113b
MD5 ad71cb4efcc5279038f671dd7a35f0f1
BLAKE2b-256 9d764c2ffc36b45f616a69819edf4876ede6ec47dc3835b2c31d78e7991eece1

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