Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/numberoverzero/accordian

Event dispatch in Python 3.8 using asyncio
https://github.com/numberoverzero/accordian

Last synced: 17 days ago
JSON representation

Event dispatch in Python 3.8 using asyncio

Awesome Lists containing this project

README

        

.. image:: https://img.shields.io/travis/numberoverzero/accordian/master.svg?style=flat-square
:target: https://travis-ci.org/numberoverzero/accordian
.. image:: https://img.shields.io/pypi/v/accordian.svg?style=flat-square
:target: https://pypi.python.org/pypi/accordian
.. image:: https://img.shields.io/github/issues-raw/numberoverzero/accordian.svg?style=flat-square
:target: https://github.com/numberoverzero/accordian/issues

Event dispatch in Python 3.8 using asyncio

Installation
------------

``pip install accordian``

Getting Started
---------------
::

import asyncio
from accordian import signal
my_event = signal("my_event")

@my_event.connect
async def pow(x, y):
await asyncio.sleep(0.1)
print(f"pow({x}, {y})")
return x ** y

@my_event.connect
async def diff(x, y):
await asyncio.sleep(0.2)
print(f"diff({x}, {y})")
return max(x, y) - min(x, y)

# create events without blocking
async def create():
my_event.send(4, 6)
my_event.send(0, 1)
asyncio.run(create())

# or block and collect results from all receivers
async def collect():
results = await my_event.join(10, 3)
assert set(results) == {1000, 7}
asyncio.run(collect())

# signals without receivers return no results
async def empty():
sig = signal("unknown")
results = await sig.join(1, "foo", None)
assert results == []
asyncio.run(empty())

Namespaces
==========

By default, ``accordian.signal`` creates signals in a global namespace. You can create your own namespaces to
group signals together. Here, a processor is passed the region and stage to create deployment tasks::

from accordian import Namespace
regions = {"east": Namespace(), "west": Namespace()}

@regions["east"].signal("dev").connect
async def deploy_east_dev(s3_url, creds):
...

@regions["east"].signal("prod").connect
async def deploy_east_prod(s3_url, creds):
# remove pre-prod feature flags
await sanitize_prod(s3_url, "east")
...

@regions["west"].signal("prod").connect
async def deploy_west_prod(s3_url, creds):
# legacy region shims
await patch_west_bundle(s3_url)
await sanitize_prod(s3_url, "west")
...

async def deploy(region, stage):
s3_url = await bundle_for_region(region, stage)
creds = await creds_for_region(region, stage)
signal = regions[region].signal(stage)

# create the deployment task without waiting
signal.send(s3_url, creds)

# create deployment tasks
asyncio.run(deploy("east", "dev"))
asyncio.run(deploy("west", "prod"))

# wait for deployments to complete
async def wait_for_tasks():
running = asyncio.all_tasks()
await asyncio.wait(running)
asyncio.run(wait_for_tasks())

Contributing
------------

Contributions welcome! Please make sure ``tox`` passes before submitting a PR.

Development
-----------

To set up a virtualenv and run the test suite::

git clone https://github.com/numberoverzero/accordian.git
make venv
make