diff --git a/keps/74-support-argo-workflow/README.md b/keps/74-support-argo-workflow/README.md index ce32c3af3d..bcba916488 100644 --- a/keps/74-support-argo-workflow/README.md +++ b/keps/74-support-argo-workflow/README.md @@ -1,4 +1,4 @@ -# KEP-74: Support Argo Workflow +# KEP-74: Support Generic Workflows - [Summary](#summary) @@ -9,56 +9,54 @@ - [User Stories](#user-stories) - [Story 1](#story-1) - [Story 2](#story-2) + - [Story 3](#story-3) + - [Story 4](#story-4) - [Design Details](#design-details) - [Workflow as An Unit](#workflow-as-an-unit) + - [How to suspend the whole workflow](#how-to-suspend-the-whole-workflow) - [Drawback and Limitations](#drawback-and-limitations) - [Advantages](#advantages) - - [Layer as An Unit](#layer-as-an-unit) - - [Examples](#examples) - - [Example 1 (ParallelSteps Contains Leaf Template Only)](#example-1-parallelsteps-contains-leaf-template-only) - - [Example 2 (ParallelSteps Contains Leaf Template and Step)](#example-2-parallelsteps-contains-leaf-template-and-step) - - [Example 3 (Workflow with Single Container Template)](#example-3-workflow-with-single-container-template) - - [How to suspend a workflow step by step](#how-to-suspend-a-workflow-step-by-step) - - [Drawback and Limitations](#drawback-and-limitations-1) + - [Stage as An Unit](#stage-as-an-unit) + - [Workflow Framework in Kueue](#workflow-framework-in-kueue) + - [How to suspend by stages](#how-to-suspend-by-stages) - [Advantages](#advantages-1) - - [Plain Pod as An Unit](#plain-pod-as-an-unit) - - [Drawback and Limitations](#drawback-and-limitations-2) + - [Drawback and Limitations](#drawback-and-limitations-1) + - [Task as An Unit](#task-as-an-unit) - [Advantages](#advantages-2) + - [Drawback and Limitations](#drawback-and-limitations-2) - [Additional Details](#additional-details) - [Test Plan](#test-plan) - [Unit Tests](#unit-tests) - [Integration tests](#integration-tests) - [Graduation Criteria](#graduation-criteria) -- [Implementation History](#implementation-history) -- [Drawbacks](#drawbacks) -- [Alternatives](#alternatives) ## Summary -This KEP outlines the proposal to integrate Argo Workflows within Kueue, discussing the advantages +This KEP outlines the proposal to integrate Generic Workflows within Kueue, discussing the advantages and disadvantages of queuing workflows at varying granularity levels, alongside detailing -the integration methodologies. +the integration methodologies. Workflow managers like Argo Workflow and Tekton can follow the convention +to queue the workflows. ## Motivation Workflows are pivotal components in domains like Scientific Computing and Simulations, where administrators often enforce resource usage quotas for different users or departments. Currently, -Argo Workflows lacks native support within Kueue. +Workflows lacks native support within Kueue. ### Goals -- Enable support for Argo Workflow within Kueue, allowing users to simply add a label +- Enable support for Generic Workflows within Kueue, allowing users to simply add a label `kueue.x-k8s.io/queue-name` to their workflows and submit them initially in a suspended state. -- Should be easily extended to support other workflow managers. +- Should be easily extended to support verious workflow managers. ### Non-Goals - +- KEP should not only work for Argo Workflow. +- Support resource reuse across multiple stages (potentially discussed in another KEP). ## Proposal - ### User Stories #### Story 1 @@ -74,39 +72,11 @@ As an ML engineer, my workflow consists of several GPU-dependent stages with uni requirements. I aim to recycle resources allocated to earlier workflow stages to boost efficiency and resource utilization. -## Design Details - -### Workflow as An Unit - -Given the diverse resource, node affinity, and toleration requirements among workflow pods, -determining the necessary resources for each flavor becomes challenging for the controller. -Users must specify workflow resources via annotations like kueue.k8s.io/max-resources, tolerations -with kueue.k8s.io/toleration, and node selectors with kueue.k8s.io/node-selector. - -#### Drawback and Limitations - -- Inability to set distinct nodeSelectors and tolerations for multiple pod sets within a workflow. - -#### Advantages +#### Story 3 -- Simplified architecture facilitating straightforward implementation. - -### Layer as An Unit - -A workflow's template definition can be a container invocation (leaf template) or a list -of steps. For workflows composed of a single leaf template, a single workload is generated. - -#### Examples - -In the following example, we solely discuss which patterns of workflows should warrant the -creation of workloads, without delving into the specifics of how these workloads are created, -nor addressing the division of responsibilities between the workflow-controller and kueue. - -##### Example 1 (ParallelSteps Contains Leaf Template Only) -For a parallelStep with only leaf templates, we create a workload for the parallelStep. -In the following example, we create workloads for `loop-example-depth-2(0:depth-1-1)` and -`loop-example-depth-2(1:depth-1-2)`. Patterns of DAGs are similar, so we do not discuss them -separately. +I am a Argo Workflow user. Templates are the basic unit to build the workflows. +Every templates can be a leaf template, step definition or DAG definition. +Example of a worflow looks like: ``` # kubectl create -f - << EOF @@ -159,11 +129,9 @@ STEP TEMPLATE PODNAME └─✔ print-message-loop(1:goodbye world) print-message loops-mlr6m-print-message-2893948292 6s ``` -##### Example 2 (ParallelSteps Contains Leaf Template and Step) +The template `print-message` is a leaf template, while the template `loop-example-depth-1` and `loop-example-depth-2` is a step definition. -For the step composed by a leaf template and another step, we create workload for the -leaf template. And the workload for the other step is created separately. -In the following example, we will create workload for `loops-644ch` and `loop-example-depth-2-2`. +Besides, step definition can also be a mix of leaf templates and other step definitions like the following example. ``` apiVersion: argoproj.io/v1alpha1 @@ -218,73 +186,399 @@ STEP TEMPLATE PODNAME └─✔ print-message(1:depth-1-2) print-message loops-644ch-print-message-3356863351 5s ``` -##### Example 3 (Workflow with Single Container Template) +The template `loop-example-depth-1` contains a step definition `loop-example-depth-2-2` and a leaf template `print-message`. + +DAG definition is similar to the step definition. + +#### Story 4 + +I amd a Tekton user. Pipelines are used to define the workflows. Pipelines contains multiple Tasks. These tasks are executed in parallel unless you set the runAfter field or finally field.Tasks contains multiple Steps. To add Steps to a Task you define a steps field (required) containing a list of desired Steps. The order in which the Steps appear in this list is the order in which they will execute. PipelineRun is the execution instance of a pipeline and TaskRun is the execution instance of a task. + +Apart from taskRef and taskSpec, pipelineRef and pipelineSpec allows you to specify a pipeline in pipelineTask. This allows you to generate a child pipelineRun which is inherited by the parent pipelineRun. + +An example of a pipeline looks like: -We create a workload for the single container template. For example: ``` -apiVersion: argoproj.io/v1alpha1 -kind: Workflow +--- +apiVersion: tekton.dev/v1 +kind: Task metadata: - generateName: hello- + name: print-result spec: - entrypoint: main - templates: - - name: main - plugin: - hello: { } + description: >- + Prints a result from another task + params: + - name: TO_PRINT + type: string + steps: + - name: print-result + image: mirror.gcr.io/bash + env: + - name: PARAM_TO_PRINT + value: $(params.TO_PRINT) + script: | + #!/usr/bin/env bash + set -e + echo $PARAM_TO_PRINT +--- +apiVersion: tekton.dev/v1 +kind: Task +metadata: + name: generate-result +spec: + description: >- + Creates strings of length based on parameters and puts them into results fields + params: + - name: STRING_LENGTH + description: Length of the string to create + - name: STRING_CHAR + description: Char to use when creating string + type: string + default: '.' + results: + - name: RESULT_STRING + description: A result string + steps: + - name: gen-result + image: mirror.gcr.io/bash + env: + - name: PARAM_STRING_LENGTH + value: $(params.STRING_LENGTH) + - name: PARAM_STRING_CHAR + value: $(params.STRING_CHAR) + script: | + #!/usr/bin/env bash + set -e + len=$PARAM_STRING_LENGTH + ch=$PARAM_STRING_CHAR + printf '%*s' "$len" | tr ' ' "$ch" >> $(results.RESULT_STRING.path) +--- +apiVersion: tekton.dev/v1 +kind: Pipeline +metadata: + name: result-test +spec: + description: >- + Generate a result of a certain length in a task and print the result in another task + params: + - name: RESULT_STRING_LENGTH + description: Length of string to generate for generate-result task + - name: RESULT_STRING_CHAR + description: Char to repeat in result string + default: '.' + tasks: + - name: generate-result + params: + - name: STRING_LENGTH + value: $(params.RESULT_STRING_LENGTH) + - name: STRING_CHAR + value: $(params.RESULT_STRING_CHAR) + taskRef: + kind: Task + name: generate-result + - name: print-result + params: + - name: TO_PRINT + value: $(tasks.generate-result.results.RESULT_STRING) + taskRef: + kind: Task + name: print-result +--- +apiVersion: tekton.dev/v1 +kind: PipelineRun +metadata: + name: result-test-run +spec: + pipelineRef: + name: result-test + params: + - name: RESULT_STRING_LENGTH + value: "3000" +``` -# argo get hello-jtlcw -... -STEP TEMPLATE PODNAME DURATION MESSAGE - ◷ hello-jtlcw main +## Design Details + +A generic workflow can be abstracted into a DAG like the following, where each node represents a step that requires containers invocation and executing operations. Each node may contains multiple pods, and pods executed in parallel can belong to different nodes. + +![DAG](workflow.drawio.svg) + +In this section, we will discuss the advantages and disadvantages of queuing Workflows, Stages, and Tasks individually. Additionally, we will explore how each of these can be implemented within Kueue. + +### Workflow as An Unit + +#### How to suspend the whole workflow + +Given the diverse resource, node affinity, and toleration requirements among workflow pods, +determining the necessary resources for each flavor becomes challenging for the controller. +Users must specify workflow resources via annotations like kueue.k8s.io/max-resources, tolerations +with kueue.k8s.io/toleration, and node selectors with kueue.k8s.io/node-selector. + +In this way, only one workload will be create for one workflow, so workflows can be integrated into +Kueue with current job framework. Only need to implement for the specific workflow types. + +#### Drawback and Limitations + +- Inability to set distinct nodeSelectors and tolerations for multiple pod sets within a workflow. + +#### Advantages + +- Simplified architecture facilitating straightforward implementation. Workload managers can +follow the same api to queue their workflows. + +### Stage as An Unit + +#### Workflow Framework in Kueue + +When we regard stage as an unit, multiple workloads will be created for one workflow. So the current +job framework is not suitable for this case. We need to create a new `workflow framework` for the +workflows. The generic workflow framework reconcile workflows in following way: + +``` go +type GenericWorkflow interface { + GetActiveStages() []GenericJob +} + + +func (r *workflowReconciler)ReconcileGenericWorkflow(ctx context.Context, req ctrl.Request, wf GenericWorkflow) { + dropFinalizers := GetWorkflowInstance() + if dropFinalizers { + DropFinalizers() + return + } + + // currently we do not support parent-workload for workflow + // stage should be implemented as GenericJob + stages := wf.GetActiveStages() + for _, stage := range stages { + // 1. make sure there is only a single existing instance of the workload. + // If there's no workload exists and stage is unsuspended, we'll stop it immediately. + // Stage should support suspend and resume + + // 1.1 If the workload is pending deletion, suspend the stage if needed + // and drop the finalizer. + + // 2. handle stage is finished. + + // 3. handle workload is nil. + + // 4. update reclaimable counts if implemented by the stage + + // 5. handle WaitForPodsReady only for a standalone job. + // handle a job when waitForPodsReady is enabled, and it is the main job + + // 6. handle eviction + + // 7. handle job is suspended. + + // 8. handle job is unsuspended. + r.ReconcileStage(ctx, req, stage, wf) + } +} ``` -#### How to suspend a workflow step by step +Each workflow managers can implement their own controller to integrate with the workflow framework. -It is hard for users to add suspend template manually in a workflow before each -leaf template. Some tools are needed to help them to do this. -We introduce three ways to manage the workflow. Responsebilities are different for the -workflow-controller and kueue-controller in two ways. +#### How to suspend by stages -1. Give users a CLI to help users modifying workflows to add a specific suspend template for each step. +In my daily work, I primarily encounter workflow managers in the form of Argo and Tekton. Consequently, this section will focus on these two. Other workflow management systems can draw upon the discussions in this section for implementation guidance. + +These two workflow managers support to suspend a stage before the stage start to run. But they do +not support to suspend a running stage (see https://github.com/tektoncd/pipeline/discussions/5353). +So only after the workflows support suspending a running stage, we can evict and reclaim the pod +of the workflows. Eviction and reclain of stages can be implemented in the future. + +We have three ways to suspend before starting a stage. + +Option 1. Give users a CLI to help users modifying workflows to add a specific suspend template for each step. When the workflows are suspended on this special suspend template, the job-controller in Kueue create workloads for the next step. Modification of workflow-controller is not needed for this way, so that it is easy to iterate, and no need to manage the version of argo and kueue. By in this way, users can modify their workflows to skip waiting in kueue, which maybe is not acceptable for some users. -2. Integrated Suspend Capability: We propose introducing a new specification field within workflows, such -as suspendBySteps. If workflow.spec.suspendBySteps is set to true, the workflow-controller automatically +![](suspend-with-cli.drawio.svg) + +For example, if users want to queue the workflow below: +``` yaml +apiVersion: argoproj.io/v1alpha1 +kind: Workflow +metadata: + generateName: loops- + namespace: argo +spec: + entrypoint: loop-example-depth-1 + templates: + - name: loop-example-depth-2 + steps: + - - name: print-message-loop + template: print-message + arguments: + parameters: + - name: message + value: "{{item}}" + withItems: # invoke print-message once for each item in parallel + - hello world # item 1 + - goodbye world # item 2 + - name: loop-example-depth-1 + steps: + - - name: loop-example-depth-2 + template: loop-example-depth-2 + withItems: + - depth-1-1 + - depth-1-2 + - name: loop-example-depth-2-2 + template: loop-example-depth-2 + - name: print-message + inputs: + parameters: + - name: message + container: + image: busybox + command: [sleep] + args: ["3m"] +``` + +After they call the CLI, the new workflow will be like: +``` yaml +apiVersion: argoproj.io/v1alpha1 +kind: Workflow +metadata: + generateName: loops- + namespace: argo +spec: + entrypoint: loop-example-depth-1 + templates: + - name: kueue-suspend + suspend: {} + - name: loop-example-depth-2 + steps: + - - name: suspend + template: kueue-suspend + - - name: print-message-loop + template: print-message + arguments: + parameters: + - name: message + value: "{{item}}" + withItems: # invoke print-message once for each item in parallel + - hello world # item 1 + - goodbye world # item 2 + - name: loop-example-depth-1 + steps: + - - name: loop-example-depth-2 + template: loop-example-depth-2 + withItems: + - depth-1-1 + - depth-1-2 + - name: print-message + inputs: + parameters: + - name: message + container: + image: busybox + command: [sleep] + args: ["3m"] +``` + +Option 2. Integrated Suspend Capability: We propose introducing a new specification field within workflows, such +as suspendBySteps. If `workflow.spec.suspendBySteps` is set to true, the workflow-controller automatically injects a special suspend template into each stepGroup. The kueue's job-controller monitors these and generates workloads for the following steps. Upon workload admission, the suspension step is marked as -completed. +completed. This work can not complete with the help of workflow managers. + +![](suspend-with-workflow-support.drawio.svg) -3. (Recommended) Kueue Webhook Enhancement: A new webhook is added within Kueue to intercept pod creations in the cluster. +For example, users can submit a argo workflow like this to queue the workflow: +``` yaml +apiVersion: argoproj.io/v1alpha1 +kind: Workflow +metadata: + generateName: loops- + namespace: argo +spec: + # support suspend before any step starting + suspendBySteps: true + entrypoint: loop-example-depth-1 + templates: + - name: loop-example-depth-2 + steps: + - - name: print-message-loop + template: print-message + arguments: + parameters: + - name: message + value: "{{item}}" + withItems: # invoke print-message once for each item in parallel + - hello world # item 1 + - goodbye world # item 2 + - name: loop-example-depth-1 + steps: + - - name: loop-example-depth-2 + template: loop-example-depth-2 + withItems: + - depth-1-1 + - depth-1-2 + - name: loop-example-depth-2-2 + template: loop-example-depth-2 + - name: print-message + inputs: + parameters: + - name: message + container: + image: busybox + command: [sleep] + args: ["3m"] +``` + +Option 3. Kueue Webhook Enhancement: A new webhook is added within Kueue to intercept pod creations in the cluster. This webhook verifies if the incoming pods are governed by a workflow and if the workflow carries the label `kueue.x-k8s.io/queue-name`. When these conditions are met, scheduling gates are appended to the pods. The job-controller in Kueue subsequently organizes these pods into groups (identifiable within the workflow's status) and creates corresponding workloads for each group. Following workload acceptance, the scheduling gates are removed from the pods, enabling their scheduling and execution. +Pod controller should be changed to ignore the pods governed by a queued workflow to avoid conflicts with +workflow controllers. + +![](suspend-with-webhook.drawio.svg) + +For example, if a pod governed by a queued workflow is created, the scheduling gates added to the pod will be like: +``` +apiVersion: v1 +kind: Pod +metadata: + name: test-pod +spec: + schedulingGates: + - name: kueue.x-k8s.io/admission + containers: + - name: pause + image: registry.k8s.io/pause:3.6 +``` #### Advantages -- It can support queuing by layer level. +- Gang for pods in the same stages is supported. +- Pods in different stages are queued by different workloads, so that can be push to differnent queues. + +#### Drawback and Limitations + +- Maybe need to implement logic to adapt differnet workflow managers. -### Plain Pod as An Unit +### Task as An Unit SchedulingGates are added to the pods governed by the workflow with `kueue.x-k8s.io/queue-name`. And create a separate workload for each of these pods. Enabling their scheduling and execution after workloads are admitted by Kueue. -#### Drawback and Limitations - -- Pods in same stepGroup are queued by different workload. -- Gang for stepGroup is not available. - #### Advantages - Can reuse the existing ability. +#### Drawback and Limitations + +- Pods in same stages are queued by different workloads. +- Gang for stages is not available. + ## Additional Details The implementation details will be added after the discussion. diff --git a/keps/74-support-argo-workflow/suspend-with-cli.drawio.svg b/keps/74-support-argo-workflow/suspend-with-cli.drawio.svg new file mode 100644 index 0000000000..30d93b3ac8 --- /dev/null +++ b/keps/74-support-argo-workflow/suspend-with-cli.drawio.svg @@ -0,0 +1,292 @@ + + + + + + + + + + + + + +
+
+
+ User +
+
+
+
+ + User + +
+
+ + + + +
+
+
+ new +
+ workflow +
+
+
+
+ + new... + +
+
+ + + + +
+
+
+ 1. argo build --input ${} --output ${} --action 'add-suspend' +
+
+
+
+ + 1. argo build --input ${}... + +
+
+ + + + + + +
+
+
+ origin +
+ workflow +
+
+
+
+ + origin... + +
+
+ + + + +
+
+
+ apiserver +
+
+
+
+ + apiserver + +
+
+ + + + +
+
+
+ 2. submit the workflow +
+
+
+
+ + 2. submit the workflow + +
+
+ + + + + + +
+
+
+ Workflow +
+ Manager +
+
+
+
+ + Workflow... + +
+
+ + + + + + +
+
+
+ Kueue +
+ Controller +
+
+
+
+ + Kueue... + +
+
+ + + + +
+
+
+ 3. suspend the next stage +
+
+
+
+ + 3. suspend the next... + +
+
+ + + + +
+
+
+ 6. admit the workload +
+
+
+
+ + 6. admit the workload + +
+
+ + + + + + +
+
+
+ Workflow +
+ Framework +
+
+
+
+ + Workflow... + +
+
+ + + + +
+
+
+ 4. create the workload for the stage +
+
+
+
+ + 4. create the... + +
+
+ + + + +
+
+
+ 5. resume the stage +
+
+
+
+ + 5. resume the stage + +
+
+ + + + +
+
+
+ 6. execute the stage +
+
+
+
+ + 6. execute the stage + +
+
+ + + + +
+
+
+ 7. go to step 3 +
+
+
+
+ + 7. go to step 3 + +
+
+
+ + + + + Text is not SVG - cannot display + + + +
\ No newline at end of file diff --git a/keps/74-support-argo-workflow/suspend-with-webhook.drawio.svg b/keps/74-support-argo-workflow/suspend-with-webhook.drawio.svg new file mode 100644 index 0000000000..cd3e24183e --- /dev/null +++ b/keps/74-support-argo-workflow/suspend-with-webhook.drawio.svg @@ -0,0 +1,254 @@ + + + + + + + + + + + +
+
+
+ User +
+
+
+
+ + User + +
+
+ + + + +
+
+
+ apiserver +
+
+
+
+ + apiserver + +
+
+ + + + +
+
+
+ 1. submit the workflow +
+
+
+
+ + 1. submit the workflow + +
+
+ + + + + + +
+
+
+ Workflow +
+ Manager +
+
+
+
+ + Workflow... + +
+
+ + + + + + +
+
+
+ Kueue +
+ Controller +
+
+
+
+ + Kueue... + +
+
+ + + + +
+
+
+ 2. create pod for stages +
+
+
+
+ + 2. create pod for st... + +
+
+ + + + +
+
+
+ 5. admit the workload +
+
+
+
+ + 5. admit the workload + +
+
+ + + + + + +
+
+
+ Workflow +
+ Framework +
+
+
+
+ + Workflow... + +
+
+ + + + +
+
+
+ 3. create the workload for the stage +
+
+
+
+ + 3. create the... + +
+
+ + + + +
+
+
+ 4. remove the scheduling readiness +
+
+
+
+ + 4. remove the schedul... + +
+
+ + + + +
+
+
+ 7. go to step 2 +
+
+
+
+ + 7. go to step 2 + +
+
+ + + + + + +
+
+
+ Webhook +
+
+
+
+ + Webhook + +
+
+ + + + +
+
+
+ 3. add scheduling +
+ readiness to pod +
+
+
+
+ + 3. add scheduling... + +
+
+
+ + + + + Text is not SVG - cannot display + + + +
\ No newline at end of file diff --git a/keps/74-support-argo-workflow/suspend-with-workflow-support.drawio.svg b/keps/74-support-argo-workflow/suspend-with-workflow-support.drawio.svg new file mode 100644 index 0000000000..1d8e9571af --- /dev/null +++ b/keps/74-support-argo-workflow/suspend-with-workflow-support.drawio.svg @@ -0,0 +1,233 @@ + + + + + + + + + + + +
+
+
+ User +
+
+
+
+ + User + +
+
+ + + + +
+
+
+ apiserver +
+
+
+
+ + apiserver + +
+
+ + + + +
+
+
+ 1. submit the workflow +
+
+
+
+ + 1. submit the workflow + +
+
+ + + + + + +
+
+
+ Workflow +
+ Manager +
+
+
+
+ + Workflow... + +
+
+ + + + + + +
+
+
+ Kueue +
+ Controller +
+
+
+
+ + Kueue... + +
+
+ + + + +
+
+
+ 2. suspend the next stage +
+
+
+
+ + 2. suspend the next... + +
+
+ + + + +
+
+
+ 5. admit the workload +
+
+
+
+ + 5. admit the workload + +
+
+ + + + + + +
+
+
+ Workflow +
+ Framework +
+
+
+
+ + Workflow... + +
+
+ + + + +
+
+
+ 3. create the workload for the stage +
+
+
+
+ + 3. create the... + +
+
+ + + + +
+
+
+ 4. resume the stage +
+
+
+
+ + 4. resume the stage + +
+
+ + + + +
+
+
+ 6. execute the stage +
+
+
+
+ + 6. execute the stage + +
+
+ + + + +
+
+
+ 7. go to step 2 +
+
+
+
+ + 7. go to step 2 + +
+
+
+ + + + + Text is not SVG - cannot display + + + +
\ No newline at end of file diff --git a/keps/74-support-argo-workflow/workflow.drawio.svg b/keps/74-support-argo-workflow/workflow.drawio.svg new file mode 100644 index 0000000000..ce14074df1 --- /dev/null +++ b/keps/74-support-argo-workflow/workflow.drawio.svg @@ -0,0 +1,296 @@ + + + + + + + + + + + + + + + +
+
+
+ Task +
+ A-1 +
+
+
+
+ + Task... + +
+
+ + + + +
+
+
+ Task +
+ A-2 +
+
+
+
+ + Task... + +
+
+ + + + +
+
+
+ Task +
+ A-3 +
+
+
+
+ + Task... + +
+
+ + + + +
+
+
+ Task +
+ C-1 +
+
+
+
+ + Task... + +
+
+ + + + +
+
+
+ Task +
+ C-2 +
+
+
+
+ + Task... + +
+
+ + + + +
+
+
+ Task +
+ D-1 +
+
+
+
+ + Task... + +
+
+ + + + +
+
+
+ Task +
+ D-2 +
+
+
+
+ + Task... + +
+
+ + + + + +
+
+
+ Stage A +
+
+
+
+ + Stage A + +
+
+ + + + +
+
+
+ Stage C +
+
+
+
+ + Stage C + +
+
+ + + + +
+
+
+ Stage D +
+
+
+
+ + Stage D + +
+
+ + + + + +
+
+
+ Task +
+ B-1 +
+
+
+
+ + Task... + +
+
+ + + + +
+
+
+ Stage B +
+
+
+
+ + Stage B + +
+
+ + + + + + +
+
+
+ Dep +
+
+
+
+ + Dep + +
+
+ + + + +
+
+
+ Dep +
+
+
+
+ + Dep + +
+
+ + + + +
+
+
+ Dep +
+
+
+
+ + Dep + +
+
+
+ + + + + Text is not SVG - cannot display + + + +
\ No newline at end of file