Skip to main content

A Python event framework to decouple code using events.

Project description

Eventlib-py

Eventlib-py is a simple event framework for Python that can be used to decouple your code.

Feature overview:

  • Fast - JIT-compiled event chains for fast event emission.
  • Lightweight - No extra dependencies.
  • Asynchronous - Full support for asynchronous event handlers and context managers.
  • Priority Ordering - Control the order in which event handlers are called.
  • Monitoring - Use context managers to monitor event processing.
import asyncio
import dataclasses

import eventlib


@dataclasses.dataclass
class MyEvent(eventlib.BaseEvent):
    value: str


@eventlib.subscribe()
async def on_my_event(event: MyEvent):
    print(f"Received: {event.value}")


asyncio.run(MyEvent("Hello, world!").emit_async())  # Prints: "Received: Hello, world!"

Usage

Event Inheritance

import eventlib


class MyEvent(eventlib.BaseEvent):
    pass


class SubEvent(MyEvent):
    pass


@eventlib.subscribe()
def on_base_event(event: MyEvent):
    print("Received event", event.__class__.__name__)


SubEvent().emit()  # Prints: "Received event SubEvent"

Priority Ordering

import eventlib


class MyEvent(eventlib.BaseEvent):
    pass


@eventlib.subscribe(priority=-1)
def first():
    print("first")


@eventlib.subscribe()  # default: priority = 0
def second():
    print("second")


@eventlib.subscribe(priority=1)
def third():
    print("third")


MyEvent().emit()  # Prints: "first", "second", "third"

Context Managers

import contextlib
import eventlib


class MyEvent(eventlib.BaseEvent):
    pass


@eventlib.subscribe(priority=-1000)  # Ensure that this is called first
@contextlib.contextmanager
def monitor(event: MyEvent):
    print("Event received")
    try:
        yield
    finally:
        print("Event processed")


@eventlib.subscribe()
def on_event(event: MyEvent):
    print("on_event")


MyEvent().emit()  # Prints: "Event received", "on_event", "Event processed"

Asyncio

import asyncio
import contextlib
import eventlib


class MyEvent(eventlib.BaseEvent):
    pass


@eventlib.subscribe(priority=-1000)  # Ensure that this is called first
@contextlib.asynccontextmanager
async def monitor(event: MyEvent):
    print("Event received")
    try:
        yield
    finally:
        print("Event processed")


@eventlib.subscribe()
async def async_on_event(event: MyEvent):
    print("async_on_event")


@eventlib.subscribe()
def on_event(event: MyEvent):
    print("on_event")


asyncio.run(MyEvent().emit_async())  # Prints: "Event received", "async_on_event", "on_event", "Event processed"

Benchmarks

The benchmark directory contains code to measure the performance of the eventlib-py library and compare it with a hard-coded reference implementation in Python.

Benchmark case_all

The following table shows the overhead of the eventlib-py library in the case_all benchmark. It's a mixed benchmark with all kinds of sync & async event handlers and context managers.

Quantile Hardcoded Time/Event EventLib Time/Event Overhead per Call EventLib Setup
0.50 42.289μs 45.373μs +7% 135.125μs
0.90 43.560μs 46.809μs +10% 143.000μs
0.99 46.658μs 50.048μs +15% 260.393μs

The overhead per call is the additional time that is needed to call the event handlers introduced by the eventlib-py library. The setup time is the additional nonrecurring overhead for subscribing the event handlers in the event system. It shows that in the worst case a 15% overhead per call is introduced. The expected median overhead is around 7% versus hard-coded event handling.

Development

Use poetry to setup the development environment.

poetry install --with=dev
poetry shell

Run the auto-formatter, checks and linter:

black .
isort .
mypy .
pylint .

Run the tests:

pytest --cov

Performance test:

pytest -s --runperf  tests/eventlib_tests/test_performance.py

Contributing

Contributions are welcome.

Please follow the commit convention https://www.conventionalcommits.org/en/v1.0.0/.

License

Dual-licensed under the terms of either the Apache License 2.0 or the MIT license.

SPDX-License-Identifier: (Apache-2.0 OR MIT)

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

eventlib_py-0.2.1.tar.gz (16.4 kB view details)

Uploaded Source

Built Distribution

eventlib_py-0.2.1-py3-none-any.whl (14.0 kB view details)

Uploaded Python 3

File details

Details for the file eventlib_py-0.2.1.tar.gz.

File metadata

  • Download URL: eventlib_py-0.2.1.tar.gz
  • Upload date:
  • Size: 16.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/5.1.0 CPython/3.12.5

File hashes

Hashes for eventlib_py-0.2.1.tar.gz
Algorithm Hash digest
SHA256 483649b32aecaaae47d496b2aff6fee75490541532c53f915391139c7b36d93f
MD5 c88e0f2c5a3446e0efe8468f0f5d9662
BLAKE2b-256 b848daad88f76e8cae76b566c103843a1b7651feac39c679f4ecacc76d1e3f11

See more details on using hashes here.

File details

Details for the file eventlib_py-0.2.1-py3-none-any.whl.

File metadata

  • Download URL: eventlib_py-0.2.1-py3-none-any.whl
  • Upload date:
  • Size: 14.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/5.1.0 CPython/3.12.5

File hashes

Hashes for eventlib_py-0.2.1-py3-none-any.whl
Algorithm Hash digest
SHA256 143e6322580d12dfcee8b2361f08e43b9e9cf0894d17adf7aa72616ecf5b8cc7
MD5 e169b76193912884ca52998a1be43f38
BLAKE2b-256 6469282d8fadcf440bae7ae258ed000f0d06be64d5b41b4356177039eb009c2c

See more details on using hashes here.

Supported by

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