Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Feature] Add Task Registry #329

Open
wants to merge 2 commits into
base: master
Choose a base branch
from

Conversation

funkybob
Copy link

Currently a WIP, this adds a Registry class to allow registering tasks independently of a Broker, and later register them with a given broker.

This solves the chicken-and-egg problem where decorated task functions may be imported before the configuration for the Broker has been loaded and applied.

Still needs tests...

@funkybob funkybob force-pushed the cjm/task-registry branch from ef36dce to 17f032f Compare July 14, 2020 01:19
"""
A Registry allows defining a collection of Actors not directly bound to a Broker.

This allows your code to declar Actors before configuring a Broker.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
This allows your code to declar Actors before configuring a Broker.
This allows your code to declare actors before configuring a broker.

from .actor import Actor, _queue_name_re


class Registry:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great to see this! I nearly have the same class in my projects.

I have implemented this method as well:

    def enqueue(self, message, *, delay=None):
        raise RuntimeError(
            "ActorCollector has not transferred its actors to an actual broker. "
            "Ensure transfer_actors() is called."
        )

It ensures that the user is told what is going on when trying to call send() on an actor before doing the bind.

"middleware to your Broker?"
) % (actor_name, invalid_options_list))

broker.declar_actor(actor)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In my code I have an additional helper method to allow binding multople collectors (here it would be registries) at once.

This is handy when linking up independet code from multiple packages.

def transfer_actors(broker, collectors: List[ActorCollector]):
    for ac in collectors:
        ac.transfer_actors(broker)

@ashleybartlett
Copy link

I have been able to achieve something like this by defining a middleware, that effectively looks like:

APIActors = Dict[str, Actor
API_ACTORS: APIActors = {}

class APIActionMiddleware(Middleware):
    """Middleware that exposes tasks to be executed by the API

        >>> @actor(api=True)
        ... def example():
        ...     do_stuff()

    The task can then be used in the API by whatever name you have set for this task. In the above
    example, it would be `example`. If you override the task name using `actor_name` it will use
    that. In this example, the task will be registered under `do-stuff`.

        >>> @actor(actor_name="do-stuff", api=True)
        ... def example():
        ...     do_stuff()

    Parameters:
        api(bool): Whether to allow this task to be exposed to the API. Pass this in as an extra
            option when using the `@actor(api=True)` decorator.
            Defaults to `False`.
    """

    @property
    def actor_options(self):
        return {"api"}

    def after_declare_actor(self, broker: Broker, actor: Actor) -> None:
        # pop this, otherwise it gets sent with every message which isn't necessary.
        if actor.options.pop("api", False):
            LOGGER.debug("Actor available to API %r.", actor.actor_name)
            API_ACTORS[actor.actor_name] = actor

Then I can reference API_ACTORS from anywhere in my code. Not sure if this approach works. I wanted to optionally store tasks for running via an API.

@ashleybartlett
Copy link

Ah, but I have missed the point about doing this separately from a broker. Never mind.

@AstraLuma
Copy link

flask-melodramatiq solves this as well.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants