Skip to main content

A lightweight 'Event' and 'Observable' library.

Project description

harami (ハラミ) is a lightweight "Signals" 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, adds an observer, and finally 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.widget_emitter = Observable[Widget]()

    @event(WidgetEventArgs)
    def on_widget_created(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 create_widget(color: str) -> None:
        widget = Widget(color)
        self.on_widget_created(widget)
        self.widget_emitter(widget)

class MyApp:
    def __init__(self) -> None:
        self.__widget_factory = WidgetFactory()
        self.__widget_factory.on_widget_created += self.__on_widget_created_handler
        self.__widget_factory.widget_emitter += self.__widget_observer

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

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

    def run(self) -> None:
        self.__widget_factory.create_widget('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.)
  • There are no visibility restrictions (public vs. private) for events/observables nor handlers/observers.
  • 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 via EventArgs.
  • An Event Source does not need to be parameterized, in such cases EventArgs.empty will be forwarded.
  • An EventArgs subclass does not need to be specified via @event (it defaults to EventArgs).
  • Observers can be async, even though observables do not expose an async/coro signature.
  • Observables provide a value-assignment syntax, ex: myObserver.state = 'foo', this may help simplify using observables to back properties.
  • Events offer add_handler()/remove_handler(), and observables offer attach()/detach(), each as alternatives to +=/-= syntax as seen in the example.
  • Last, but not least, Observables can 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-2.0.0.tar.gz (7.8 kB view details)

Uploaded Source

Built Distribution

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

harami-2.0.0-py3-none-any.whl (7.9 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: harami-2.0.0.tar.gz
  • Upload date:
  • Size: 7.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.11.15+

File hashes

Hashes for harami-2.0.0.tar.gz
Algorithm Hash digest
SHA256 f99377c75349d82d5b478d6d9854b31125f66110f5a806998b733aaa4ddb3f92
MD5 d1957b4fdfddc53ad4253348bb25d4cc
BLAKE2b-256 eed1b96182b3ab667e3d2a0bde938c6d6f8a4c177f271a53790c8acb9799e61c

See more details on using hashes here.

File details

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

File metadata

  • Download URL: harami-2.0.0-py3-none-any.whl
  • Upload date:
  • Size: 7.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.11.15+

File hashes

Hashes for harami-2.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 e13822430b79a3d9432a99f065f061c79b7491dd7adf0b3dceece74f97cf1065
MD5 d9069055c12624288b67c5331c93f1e5
BLAKE2b-256 cca23ef0d516cc9cdae6aa792725911ceb6fb9803fc47f819e1b04e7344d44ef

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