With version 2, disBatch consists of three major components:
- A driver (aka controller) that maintains the state of the task processing.
- An execution context that encapsulates one or more engines running on one or mode nodes. A disBatch run may have multiple contexts.
- An engine that is a collection of cylinder threads. Each cylinder runs a loop that waits for a task from the controller, spawns a sub-process to evaluate it, waits for the sub-process to exit, and then sends a report to the controller.
In normal operation, the driver spawns a couple of threads. One implements that KVS service. Another is the task feeder. This takes tasks from a task generator and hands them off to the controller via KVS.
Each task has an age, which reflects the number of synchronization events that preceded it. Synchronization events are barriers and per engine tasks. Per engine tasks are posted to KVS. A barrier is tracked by the controller. When all tasks prior to the barrier have been completed, the barrier is satisfied, a per engine event to this effect is posted to KVS and the controller's age is increased. The controller notifies the task feeder thread of the new age. The task feeder will not issue a task to the controller unless the controller's age is equal to the task's age. This interplay ensures no task is available for execution until all previous barriers (and thus in turn, all previous tasks) have been accounted for. Something akin to this takes place between an engine and its cylinders to implement per engine synchronization. See below.
The controller executes a main loop that waits for a controller event to arrive from KVS. These events include a new task from the task feeder, a completed task report from a cylinder, the registration of an execution context or an engine, a cylinder start, the notification that a context, engine or cylinder has stopped, requests to shutdown a context or a cylinder, and a few other events.
Each pass through the loop, the controller:
- Accepts a controller message from KVS. These may lead it to alter its internal state (say add a new cylinder) or execute an operation like sending a shutdown message to an engine. Of particular interest are messages providing a new task to execute, which causes that task to be added to a list of tasks with the same age, and messages reporting the completion of a task, which causes the cylinder it was assigned to to be marked available again and the finished task id to be recorded.
- Checks to see if all necessary tasks have been completed to satisfy a barrier. If so the age is advanced, and other barriers iteratively checked---that is the completion of one task could in effect satisfy a series of successive barriers.
- If there are tasks for the current age and available cylinders, assign tasks to the individual cylinders until we run out of one or the other. Note: If we record the assignments (including the full task), it should be straightforward to reissue tasks upon engine "failure".
- Update overall status info that is kept in KVS. This is used by
dbmon.py
to provide quasi-realtime info about the state of a disBatch run.
As noted, the driver receives messages informing it of new contexts, engines and cylinders. A portion of this information is incorporated in the status report placed in KVS. It is also used to implement task limits for contexts. Once the controller has assigned the cylinder(s) of the engine(s) of a context a total number of tasks equal to the task limit specified for the context, it sends a shutdown request to every engine in the context.
A context is responsible for interfacing between a collection of computational resources and a controller. Currently two kinds are supported:
- SLURM: This context makes use of environment variables set by SLURM to identify the allocated nodes and uses
srun
to start engines. The code here could serve as a model for implementing contexts for other batch queuing systems. - SSH: The nodes to be used are passed via the command line option (
-s
) or the environment variableDISBATCH_SSH_NODELIST
. Engines are started viassh
.
Each context monitors its engines and invokes a retirement method, if provided, when an engine exits.
A context is also a logical shutdown unit. The user can, for example via dbmon.py
, request that a context be shutdown. This is implemented by sending a shutdown request to each of the context's engines. Note: Such a request waits politely for all cylinders to complete any currently assigned tasks before stopping the engine.
An engine is a collection of N+1 cylinder threads, where N is the number of allowable concurrently executing tasks specified for the engine. The extra cylinder handles the per-engine tasks. Per-engine tasks are maintained as an ordered queue in KVS: engines view
values using a key with an index, stepping the index each time. Thus an engine joining at any given time can "replay" all the per engine activity. As it does so, it updates its internal age, and notifies each of its cylinders of the current age. A cylinder will not execute an assigned task until the engine has reached that task's age.
With the exception of some reporting details, the "standard" case should be the same as with version 1.
With version 2, a user can invoke disBatch
with -S
, which starts a disBatch "service"---effectively just the controller. In this case, the name of a utility script is displayed. This script (always created by version 2), can be submitted via sbatch to add an execution context. One could even submit this with a job array specification, and so add nodes on the fly to the disBatch run. The same script can be invoked with -s
to add some ssh hosts to the mix, e.g., the user's own workstation.
The script can be invoked with --mon
to start up a simple ASCII-UI to monitor progress and request shutdown of an engine or a context.
-
The controller is supposed to be the only single point of failure, nothing else (in the disBatch system) should be (assuming non malicious failure). Barriers (including an implicit one at the end), of course, might not be satisfied, but that aside a disBatch run can keep going even if a context or engine dies (if all engines died, more would have to be added to make more progress).
-
Idempotency and task reissue.
-
cli version of dbmon.py.
-
Job array demo. (Theory vs practice.)
-
Add option to insert
timeout
? -
Add heartbeat as a failure detection mechanism?
-
pernode vs perengine
-
Remove delay for explicitly started engines? Probably not ...