Skip to main content

Fast and compact framework for communication between threads and processes in Python using event loops, signals and slots.

Project description

signal-slot

tests Downloads

Qt-like event loops, signals and slots for communication across threads and processes in Python.

Installation

pip install signal-slot-mp

Linux, macOS, and Windows are supported.

Overview

signal-slot enables a parallel programming paradigm inspired by Qt's signals and slots, but in Python.

The main idea can be summarized as follows:

  • Application is a collection of EventLoops. Each EventLoop is an infinite loop that occupies a thread or a process.
  • Logic of the system is implemented in EventLoopObjects that live on EventLoops. Each EventLoop can support multiple EventLoopObjects.
  • EventLoopObjects can emit signals. A signal "message" contains a name of the signal and the payload (arbitrary data).
  • Components can also connect to signals emitted by other components by specifying a slot function to be called when the signal is received by the EventLoop.

Usage example

import time
import datetime
from signal_slot.signal_slot import EventLoop, EventLoopObject, EventLoopProcess, Timer, signal

now = datetime.datetime.now

# classes derived from EventLoopObject define signals and slots (actually any method can be a slot)
class A(EventLoopObject):
    @signal
    def signal_a(self):
        ...

    def on_signal_b(self, msg: str):
        print(f"{now()} {self.object_id} received signal_b: {msg}")
        time.sleep(1)
        self.signal_a.emit("hello from A", 42)

class B(EventLoopObject):
    @signal
    def signal_b(self):
        ...

    def on_signal_a(self, msg: str, other_data: int):
        print(f"{now()} {self.object_id} received signal_a: {msg} {other_data}")
        time.sleep(1)
        self.signal_b.emit("hello from B")

# create main event loop and object of type A
main_event_loop = EventLoop("main_loop")
a = A(main_event_loop, "object: a")

# create a background process with a separate event loop and object b that lives on that event loop
bg_process = EventLoopProcess(unique_process_name="background_process")
b = B(bg_process.event_loop, "object: b")

# connect signals and slots
a.signal_a.connect(b.on_signal_a)
b.signal_b.connect(a.on_signal_b)

# emit signal from a to kick off the communication
a.signal_a.emit("Initial hello from A", 1337)

# create a timer that will stop our system after 10 seconds
stop_timer = Timer(main_event_loop, 10.0, single_shot=True)
stop_timer.start()

# connect the stop method of the event loop to the timeout signal of the timer
stop_timer.timeout.connect(main_event_loop.stop)
stop_timer.timeout.connect(bg_process.stop)  # stops the event loop of the background process

# start the background process
bg_process.start()

# start the main event loop
main_event_loop.exec()

# if we get here, the main event loop has stopped
# wait for the background process to finish
bg_process.join()

print(f"{now()} Done!")

The output should roughly look like this:

2022-11-30 01:51:58.943425 object: b received signal_a: Initial hello from A 1337
2022-11-30 01:51:59.944957 object: a received signal_b: hello from B
2022-11-30 01:52:00.945852 object: b received signal_a: hello from A 42
2022-11-30 01:52:01.947599 object: a received signal_b: hello from B
2022-11-30 01:52:02.949214 object: b received signal_a: hello from A 42
2022-11-30 01:52:03.950762 object: a received signal_b: hello from B
2022-11-30 01:52:04.952419 object: b received signal_a: hello from A 42
2022-11-30 01:52:05.953596 object: a received signal_b: hello from B
2022-11-30 01:52:06.954918 object: b received signal_a: hello from A 42
2022-11-30 01:52:07.956701 object: a received signal_b: hello from B
2022-11-30 01:52:08.957755 object: b received signal_a: hello from A 42
2022-11-30 01:52:09.963144 Done!

Implementation details

  • There's not argument validation for signals and slots. If you connect a slot to a signal with a different signature, it will fail at runtime. This can also be used to your advantage by allowing to propagate arbitrary data as payload with appropriate runtime checks.
  • It is currently impossible to connect a slot to a signal if emitter and receiver objects belong to event loops already running in different processes (although it should be possible to implement this feature). Connect signals to slots during system initialization.
  • Signal-slot mechanism in the current implementation can't implement a message passing protocol where only a single copy of the signal is received by the subscribers. Signals are always delivered to all connected slots. Use a FIFO multiprocessing queue if you want only one receiver to receive the signal.

Multiprocessing queues

At the core of the signal-slot mechanism are the queues that are used to pass messages between processes. Python provides a default implementation multiprocessing.Queue, which turns out to be rather slow.

By default we use a custom queue implementation written in C++ using POSIX API that is significantly faster: https://github.com/alex-petrenko/faster-fifo.

Contributing

Local installation for development:

pip install -e .[dev]

Automatic code formatting:

make format && make check-codestyle

Run tests:

make test

Recent releases

v1.0.5
  • Windows support (do not require POSIX-only faster-fifo on Windows)
v1.0.4
  • Use updated version of faster-fifo
v1.0.3
  • Improved logging
v1.0.2
  • Catching queue.Full exception to handle situations where receiver event loop process is killed
v1.0.1
  • Added signal_slot.configure_logger() function to configure a custom logger
v1.0.0
  • First PyPI version

Footnote

Originally designed for Sample Factory 2.0, a high-throughput asynchronous RL codebase https://github.com/alex-petrenko/sample-factory. Distributed under MIT License (see LICENSE), feel free to use for any purpose, commercial or not, at your own risk.

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

signal-slot-mp-1.0.5.tar.gz (14.5 kB view details)

Uploaded Source

Built Distribution

signal_slot_mp-1.0.5-py3-none-any.whl (13.7 kB view details)

Uploaded Python 3

File details

Details for the file signal-slot-mp-1.0.5.tar.gz.

File metadata

  • Download URL: signal-slot-mp-1.0.5.tar.gz
  • Upload date:
  • Size: 14.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.8.16

File hashes

Hashes for signal-slot-mp-1.0.5.tar.gz
Algorithm Hash digest
SHA256 61c99044ac1ffbc01e5d6edebaa6294bb1346efe54cf195f01cfc898a95dd915
MD5 138c33c76525e01014f093d5914e46c6
BLAKE2b-256 3f2a86bcc1df3132b7b3105be124299946cd053851593d6a8ab34a822a1ec65f

See more details on using hashes here.

File details

Details for the file signal_slot_mp-1.0.5-py3-none-any.whl.

File metadata

File hashes

Hashes for signal_slot_mp-1.0.5-py3-none-any.whl
Algorithm Hash digest
SHA256 aa069ccaf408561271d2c455d60afd01066bb015a9c5b8af1d350a6722c56c68
MD5 d3b2fc5b32d3be4a5b86d3ed42e6a382
BLAKE2b-256 f5d4f886f72762a9fe465edf952662747fb595e2b4ab2f9fe948bbc27ba41055

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