Skip to main content

A lightweight 'Event' and 'Observable' library.

Project description

harami is a lightweight "Event" and "Observable" library for Python.

This README is only a high-level introduction to harami. For more detailed documentation, please view the official docs at https://harami.readthedocs.io.

Installation

harami can be installed from pypi through the usual means:

pip install harami

Usage

Let's try a "learn by example" approach. The following code is a Widget Factory implementation. The Widget Factory raises an event on Widget Creation, it also emits the Widgets it creates through an Observable. The MyApp class creates an instance of the Widget Factory, adds an event handler, and also adds an observer, then requests the factory create a Widget.

from harami import *

class Widget:
    color:str
    def __init__(self, color:str = None) -> None:
        self.color = color

class WidgetEventArgs(EventArgs):
    @property
    def widget(self) -> Widget:
        return self.args[0]

class WidgetFactory:
    def __init__(self) -> None:
        self.widgetEmitter = Observable[Widget]()

    @event
    def onWidgetCreated(self, widget:Widget) -> None:
        # this is an "Event Source", it does not require an
        # implementation, but one can be provided if it
        # makes sense for your application
        pass

    def createWidget(color:str) -> None:
        widget = Widget(color)
        self.onWidgetCreated(widget)
        self.widgetEmitter(widget)

class MyApp:
    def __init__(self) -> None:
        self.__widgetFactory = WidgetFactory()
        self.__widgetFactory.onWidgetCreated += self.__onWidgetCreatedHandler
        self.__widgetFactory.widgetEmitter += self.__widgetObserver

    def __onWidgetCreatedHandler(sender:object, e:WidgetEventArgs) -> None:
        print(f'Widget Created: color is {e.widget.color}')

    def __widgetObserver(widget:Widget) -> None:
        print(f'Widget Emitted: color is {e.widget.color}')

    def run(self) -> None:
        self.__eventExample.createWidget('red')

MyApp().run()

When executed the program outputs the following:

Widget Created: color is red
Widget Emitted: color is red

Notables..

Things not obvious given the example above:

  • Events and Observables can be declared at a module scope (classes are not required.)
  • The names you use for events, observables, handlers, and observers are not relevant, name things as you please.
  • The scoping (public vs. private) of members and functions is not important, scope as you please.
  • Event Handlers can be async, whether or not the Event Source is async.
  • Event Sources can be async, whether or not event handlers are async.
  • Event Handlers receive an EventArgs, which you can optionally subclass as seen in the example.
  • All *args and *kwargs passed to an Event Source are forwarded via an args attribute of type tuple and a kwargs attribute of type dict, both accessible on EventArgs and any EventArgs subclass. You can use these in the raw for downstream method invocations or wrap them behind properties as seen in the example above.
  • Observers can be async, even though observables do not expose an async/coro signature.
  • Observables provide a value-assignment syntax, ex: myObserver.state = 'foo' if that is preferred, this also simplifies using an observable to back properties.
  • Events offer addHandler() and removeHandler(), observables offer attach() and detach(), each as respective alternatives to the += and -= shorthand syntax seen in the example.
  • Last, but not least, Observables and be used as Event Handlers, and Event Sources can be used as Observers.

This library is meant to be lightweight and not have dependencies on other libraries, as such it has an intentionally narrow focus.

Contact

You can reach me on Discord or open an Issue on Github.

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

harami-0.0.2.tar.gz (5.7 kB view details)

Uploaded Source

Built Distribution

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

harami-0.0.2-py3-none-any.whl (6.5 kB view details)

Uploaded Python 3

File details

Details for the file harami-0.0.2.tar.gz.

File metadata

  • Download URL: harami-0.0.2.tar.gz
  • Upload date:
  • Size: 5.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.3+

File hashes

Hashes for harami-0.0.2.tar.gz
Algorithm Hash digest
SHA256 f99a75fab542adc7418b086b750d67915b8c940d498f6faf7b99cd4897a334d4
MD5 a579d047818e3a0de7da3fddaaaf0e6b
BLAKE2b-256 58c9fb733d74f29792faa1c22f8f97d2f7fe2ecbb1ded0de815bb677d92ea5b7

See more details on using hashes here.

File details

Details for the file harami-0.0.2-py3-none-any.whl.

File metadata

  • Download URL: harami-0.0.2-py3-none-any.whl
  • Upload date:
  • Size: 6.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.3+

File hashes

Hashes for harami-0.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 8c94e4f2b1c7585974f69467c72622ae6e54c587c4d35116b3d31c7a11b01232
MD5 006664dcfcd4b760a3217e111bee4f83
BLAKE2b-256 dbccde66dca4094796983ae5fe79d6a402f4dc6c8b87efd697d328cff3936196

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