Merge "Add docs for the worker based engine (WBE)"
This commit is contained in:
commit
162c683397
@ -99,22 +99,8 @@ Worker-Based
|
||||
|
||||
**Engine type**: ``'worker-based'``
|
||||
|
||||
This is engine that schedules tasks to **workers** -- separate processes
|
||||
dedicated for certain tasks execution, possibly running on other machines,
|
||||
connected via `amqp <http://www.amqp.org/>`_ (or other supported
|
||||
`kombu <http://kombu.readthedocs.org/>`_ transports). For more information,
|
||||
please see `wiki page`_ for more details on how the worker based engine
|
||||
operates.
|
||||
|
||||
.. note::
|
||||
|
||||
This engine is under active development and is experimental but it is
|
||||
usable and does work but is missing some features (please check the
|
||||
`blueprint page`_ for known issues and plans) that will make it more
|
||||
production ready.
|
||||
|
||||
.. _wiki page: https://wiki.openstack.org/wiki/TaskFlow/Worker-based_Engine
|
||||
.. _blueprint page: https://blueprints.launchpad.net/taskflow
|
||||
For more information, please see :doc:`workers <workers>` for more details on
|
||||
how the worker based engine operates (and the design decisions behind it).
|
||||
|
||||
Interfaces
|
||||
==========
|
||||
|
BIN
doc/source/img/distributed_flow_rpc.png
Normal file
BIN
doc/source/img/distributed_flow_rpc.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 67 KiB |
@ -14,6 +14,7 @@ Contents
|
||||
arguments_and_results
|
||||
patterns
|
||||
engines
|
||||
workers
|
||||
jobs
|
||||
inputs_and_outputs
|
||||
notifications
|
||||
@ -23,7 +24,6 @@ Contents
|
||||
utils
|
||||
states
|
||||
|
||||
|
||||
Indices and tables
|
||||
==================
|
||||
|
||||
|
328
doc/source/workers.rst
Normal file
328
doc/source/workers.rst
Normal file
@ -0,0 +1,328 @@
|
||||
-------
|
||||
Workers
|
||||
-------
|
||||
|
||||
Overview
|
||||
========
|
||||
|
||||
This is engine that schedules tasks to **workers** -- separate processes
|
||||
dedicated for certain atoms execution, possibly running on other machines,
|
||||
connected via `amqp`_ (or other supported
|
||||
`kombu <http://kombu.readthedocs.org/>`_ transports).
|
||||
|
||||
.. note::
|
||||
|
||||
This engine is under active development and is experimental but it is
|
||||
usable and does work but is missing some features (please check the
|
||||
`blueprint page`_ for known issues and plans) that will make it more
|
||||
production ready.
|
||||
|
||||
.. _blueprint page: https://blueprints.launchpad.net/taskflow?searchtext=wbe
|
||||
|
||||
Terminology
|
||||
-----------
|
||||
|
||||
Client
|
||||
Code or program or service that uses this library to define flows and
|
||||
run them via engines.
|
||||
|
||||
Transport + protocol
|
||||
Mechanism (and `protocol`_ on top of that mechanism) used to pass information
|
||||
between the client and worker (for example amqp as a transport and a json
|
||||
encoded message format as the protocol).
|
||||
|
||||
Executor
|
||||
Part of the worker-based engine and is used to publish task requests, so
|
||||
these requests can be accepted and processed by remote workers.
|
||||
|
||||
Worker
|
||||
Workers are started on remote hosts and has list of tasks it can perform (on
|
||||
request). Workers accept and process task requests that are published by an
|
||||
executor. Several requests can be processed simultaneously in separate
|
||||
threads. For example, an `executor`_ can be passed to the worker and
|
||||
configured to run in as many threads (green or not) as desired.
|
||||
|
||||
Proxy
|
||||
Executors interact with workers via a proxy. The proxy maintains the underlying
|
||||
transport and publishes messages (and invokes callbacks on message reception).
|
||||
|
||||
Requirements
|
||||
------------
|
||||
|
||||
* **Transparent:** it should work as ad-hoc replacement for existing
|
||||
*(local)* engines with minimal, if any refactoring (e.g. it should be
|
||||
possible to run the same flows on it without changing client code if
|
||||
everything is set up and configured properly).
|
||||
* **Transport-agnostic:** the means of transport should be abstracted so that
|
||||
we can use `oslo.messaging`_, `gearmand`_, `amqp`_, `zookeeper`_, `marconi`_,
|
||||
`websockets`_ or anything else that allows for passing information between a
|
||||
client and a worker.
|
||||
* **Simple:** it should be simple to write and deploy.
|
||||
* **Non-uniformity:** it should support non-uniform workers which allows
|
||||
different workers to execute different sets of atoms depending on the workers
|
||||
published capabilities.
|
||||
|
||||
.. _marconi: https://wiki.openstack.org/wiki/Marconi
|
||||
.. _zookeeper: http://zookeeper.org/
|
||||
.. _gearmand: http://gearman.org/
|
||||
.. _oslo.messaging: https://wiki.openstack.org/wiki/Oslo/Messaging
|
||||
.. _websockets: http://en.wikipedia.org/wiki/WebSocket
|
||||
.. _amqp: http://www.amqp.org/
|
||||
.. _executor: https://docs.python.org/dev/library/concurrent.futures.html#executor-objects
|
||||
.. _protocol: http://en.wikipedia.org/wiki/Communications_protocol
|
||||
|
||||
Use-cases
|
||||
---------
|
||||
|
||||
* `Glance`_
|
||||
|
||||
* Image tasks *(long-running)*
|
||||
|
||||
* Convert, import/export & more...
|
||||
|
||||
* `Heat`_
|
||||
|
||||
* Engine work distribution
|
||||
|
||||
* `Rally`_
|
||||
|
||||
* Load generation
|
||||
|
||||
* *Your use-case here*
|
||||
|
||||
.. _Heat: https://wiki.openstack.org/wiki/Heat
|
||||
.. _Rally: https://wiki.openstack.org/wiki/Rally
|
||||
.. _Glance: https://wiki.openstack.org/wiki/Glance
|
||||
|
||||
Design
|
||||
======
|
||||
|
||||
There are two communication sides, the *executor* and *worker* that communicate
|
||||
using a proxy component. The proxy is designed to accept/publish messages
|
||||
from/into a named exchange.
|
||||
|
||||
High level architecture
|
||||
-----------------------
|
||||
|
||||
.. image:: img/distributed_flow_rpc.png
|
||||
:height: 275px
|
||||
:align: right
|
||||
|
||||
Executor and worker communication
|
||||
---------------------------------
|
||||
|
||||
Let's consider how communication between an executor and a worker happens.
|
||||
First of all an engine resolves all atoms dependencies and schedules atoms that
|
||||
can be performed at the moment. This uses the same scheduling and dependency
|
||||
resolution logic that is used for every other engine type. Then the atoms which
|
||||
can be executed immediately (ones that are dependent on outputs of other tasks
|
||||
will be executed when that output is ready) are executed by the worker-based
|
||||
engine executor in the following manner:
|
||||
|
||||
1. The executor initiates task execution/reversion using a proxy object.
|
||||
2. :py:class:`~taskflow.engines.worker_based.proxy.Proxy` publishes task
|
||||
request (format is described below) into a named exchange using a routing
|
||||
key that is used to deliver request to particular workers topic. The executor
|
||||
then waits for the task requests to be accepted and confirmed by workers. If
|
||||
the executor doesn't get a task confirmation from workers within the given
|
||||
timeout the task is considered as timed-out and a timeout exception is
|
||||
raised.
|
||||
3. A worker receives a request message and starts a new thread for processing it.
|
||||
|
||||
1. The worker dispatches the request (gets desired endpoint that actually
|
||||
executes the task).
|
||||
2. If dispatched succeeded then the worker sends a confirmation response
|
||||
to the executor otherwise the worker sends a failed response along with
|
||||
a serialized :py:class:`failure <taskflow.utils.misc.Failure>` object
|
||||
that contains what has failed (and why).
|
||||
3. The worker executes the task and once it is finished sends the result
|
||||
back to the originating executor (every time a task progress event is
|
||||
triggered it sends progress notification to the executor where it is
|
||||
handled by the engine, dispatching to listeners and so-on).
|
||||
|
||||
4. The executor gets the task request confirmation from the worker and the task
|
||||
request state changes from the ``PENDING`` to the ``RUNNING`` state. Once
|
||||
a task request is in the ``RUNNING`` state it can't be timed-out (considering
|
||||
that task execution process may take unpredictable time).
|
||||
5. The executor gets the task execution result from the worker and passes it
|
||||
back to the executor and worker-based engine to finish task processing (this
|
||||
repeats for subsequent tasks).
|
||||
|
||||
.. note::
|
||||
|
||||
:py:class:`~taskflow.utils.misc.Failure` objects are not json-serializable
|
||||
(they contain references to tracebacks which are not serializable), so they
|
||||
are converted to dicts before sending and converted from dicts after
|
||||
receiving on both executor & worker sides (this translation is lossy since
|
||||
the traceback won't be fully retained).
|
||||
|
||||
Executor request format
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
* **task** - full task name to be performed
|
||||
* **action** - task action to be performed (e.g. execute, revert)
|
||||
* **arguments** - arguments the task action to be called with
|
||||
* **result** - task execution result (result or
|
||||
:py:class:`~taskflow.utils.misc.Failure`) *[passed to revert only]*
|
||||
|
||||
Additionally, the following parameters are added to the request message:
|
||||
|
||||
* **reply_to** - executor named exchange workers will send responses back to
|
||||
* **correlation_id** - executor request id (since there can be multiple request
|
||||
being processed simultaneously)
|
||||
|
||||
**Example:**
|
||||
|
||||
.. code:: json
|
||||
|
||||
{
|
||||
"action": "execute",
|
||||
"arguments": {
|
||||
"joe_number": 444
|
||||
},
|
||||
"task": "tasks.CallJoe"
|
||||
}
|
||||
|
||||
Worker response format
|
||||
~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
When **running:**
|
||||
|
||||
.. code:: json
|
||||
|
||||
{
|
||||
"status": "RUNNING"
|
||||
}
|
||||
|
||||
When **progressing:**
|
||||
|
||||
.. code:: json
|
||||
|
||||
{
|
||||
"event_data": <event_data>,
|
||||
"progress": <progress>,
|
||||
"state": "PROGRESS"
|
||||
}
|
||||
|
||||
When **succeeded:**
|
||||
|
||||
.. code:: json
|
||||
|
||||
{
|
||||
"event": <event>,
|
||||
"result": <result>,
|
||||
"state": "SUCCESS"
|
||||
}
|
||||
|
||||
When **failed:**
|
||||
|
||||
.. code:: json
|
||||
|
||||
{
|
||||
"event": <event>,
|
||||
"result": <misc.Failure>,
|
||||
"state": "FAILURE"
|
||||
}
|
||||
|
||||
Usage
|
||||
=====
|
||||
|
||||
|
||||
Workers
|
||||
-------
|
||||
|
||||
To use the worker based engine a set of workers must first be established on
|
||||
remote machines. These workers must be provided a list of task objects, task
|
||||
names, modules names (or entrypoints that can be examined for valid tasks) they
|
||||
can respond to (this is done so that arbitrary code execution is not possible).
|
||||
|
||||
For complete parameters and object usage please visit
|
||||
:py:class:`~taskflow.engines.worker_based.worker.Worker`.
|
||||
|
||||
**Example:**
|
||||
|
||||
.. code:: python
|
||||
|
||||
from taskflow.engines.worker_based import worker as w
|
||||
|
||||
config = {
|
||||
'url': 'amqp://guest:guest@localhost:5672//',
|
||||
'exchange': 'test-exchange',
|
||||
'topic': 'test-tasks',
|
||||
'tasks': ['tasks:TestTask1', 'tasks:TestTask2'],
|
||||
}
|
||||
worker = w.Worker(**config)
|
||||
worker.run()
|
||||
|
||||
Engines
|
||||
-------
|
||||
|
||||
To use the worker based engine a flow must be constructed (which contains tasks
|
||||
that are visible on remote machines) and the specific worker based engine
|
||||
entrypoint must be selected. Certain configuration options must also be
|
||||
provided so that the transport backend can be configured and initialized
|
||||
correctly. Otherwise the usage should be mostly transparent (and is nearly
|
||||
identical to using any other engine type).
|
||||
|
||||
For complete parameters and object usage please see
|
||||
:py:class:`~taskflow.engines.worker_based.engine.WorkerBasedActionEngine`.
|
||||
|
||||
**Example with amqp transport:**
|
||||
|
||||
.. code:: python
|
||||
|
||||
engine_conf = {
|
||||
'engine': 'worker-based',
|
||||
'url': 'amqp://guest:guest@localhost:5672//',
|
||||
'exchange': 'test-exchange',
|
||||
'topics': ['topic1', 'topic2'],
|
||||
}
|
||||
flow = lf.Flow('simple-linear').add(...)
|
||||
eng = taskflow.engines.load(flow, engine_conf=engine_conf)
|
||||
eng.run()
|
||||
|
||||
**Example with filesystem transport:**
|
||||
|
||||
.. code:: python
|
||||
|
||||
engine_conf = {
|
||||
'engine': 'worker-based',
|
||||
'exchange': 'test-exchange',
|
||||
'topics': ['topic1', 'topic2'],
|
||||
'transport': 'filesystem',
|
||||
'transport_options': {
|
||||
'data_folder_in': '/tmp/test',
|
||||
'data_folder_out': '/tmp/test',
|
||||
},
|
||||
}
|
||||
flow = lf.Flow('simple-linear').add(...)
|
||||
eng = taskflow.engines.load(flow, engine_conf=engine_conf)
|
||||
eng.run()
|
||||
|
||||
Limitations
|
||||
===========
|
||||
|
||||
* Atoms inside a flow must receive and accept parameters only from the ways
|
||||
defined in :doc:`persistence`. In other words, the task that is created when
|
||||
a workflow is constructed will not be the same task that is executed on a
|
||||
remote worker (and any internal state not passed via the
|
||||
:doc:`inputs_and_outputs` mechanism can not be transferred). This means
|
||||
resource objects (database handles, file descriptors, sockets, ...) can **not**
|
||||
be directly sent across to remote workers (instead the configuration that
|
||||
defines how to fetch/create these objects must be instead).
|
||||
* Worker-based engines will in the future be able to run lightweight tasks
|
||||
locally to avoid transport overhead for very simple tasks (currently it will
|
||||
run even lightweight tasks remotely, which may be non-performant).
|
||||
* Fault detection, currently when a worker acknowledges a task the engine will
|
||||
wait for the task result indefinitely (a task could take a very long time to
|
||||
finish). In the future there needs to be a way to limit the duration of a
|
||||
remote workers execution (and track there liveness) and possibly spawn
|
||||
the task on a secondary worker if a timeout is reached (aka the first worker
|
||||
has died or has stopped responding).
|
||||
|
||||
Interfaces
|
||||
==========
|
||||
|
||||
.. automodule:: taskflow.engines.worker_based.worker
|
||||
.. automodule:: taskflow.engines.worker_based.engine
|
||||
.. automodule:: taskflow.engines.worker_based.proxy
|
@ -20,6 +20,20 @@ from taskflow import storage as t_storage
|
||||
|
||||
|
||||
class WorkerBasedActionEngine(engine.ActionEngine):
|
||||
"""Worker based action engine.
|
||||
|
||||
Specific backend configuration:
|
||||
|
||||
:param exchange: broker exchange exchange name in which executor / worker
|
||||
communication is performed
|
||||
:param url: broker connection url (see format in kombu documentation)
|
||||
:param topics: list of workers topics to communicate with (this will also
|
||||
be learned by listening to the notifications that workers
|
||||
emit).
|
||||
:keyword transport: transport to be used (e.g. amqp, memory, etc.)
|
||||
:keyword transport_options: transport specific options
|
||||
"""
|
||||
|
||||
_storage_cls = t_storage.SingleThreadedStorage
|
||||
|
||||
def _task_executor_cls(self):
|
||||
|
Loading…
Reference in New Issue
Block a user