Built-in Workflows

Overview

Cloudify comes with a number of built-in workflows, covering:

Built-in workflows are declared and mapped in types.yaml, which is usually imported either directly or indirectly via other imports.

# Snippet from types.yaml

workflows:
  install: default_workflows.cloudify.plugins.workflows.install
  uninstall: default_workflows.cloudify.plugins.workflows.uninstall
  execute_operation:
    mapping: default_workflows.cloudify.plugins.workflows.execute_operation
    parameters:
      operation: {}
      operation_kwargs:
        default: {}
      run_by_dependency_order:
        default: false
      type_names:
        default: []
      node_ids:
        default: []
      node_instance_ids:
        default: []

The implementations for these workflows can be found at cloudify-plugins-common.

Built-in workflows are not special in any way - they use the same API and framework as any custom workflow is able to use, and one may replace them with different workflows with the same names.

The Install Workflow

Workflow name: install

Workflow description: Workflow for installing applications.

Workflow high-level pseudo-code:

For each node, for each node instance (in parallel):

  1. Wait for node instance relationships to be started. (Only start processing this node instance when the node instances it depends on are started).
  2. Execute cloudify.interfaces.validation.create operation. 1
  3. Execute cloudify.interfaces.lifecycle.precreate operation. 1
  4. Execute cloudify.interfaces.lifecycle.create operation. 1
  5. Execute cloudify.interfaces.relationship_lifecycle.preconfigure relationship operations.2
  6. Execute cloudify.interfaces.lifecycle.configure operation.1
  7. Execute cloudify.interfaces.relationship_lifecycle.postconfigure relationship operations.2
  8. Execute cloudify.interfaces.lifecycle.start operation.1
  9. If the node instance is a host node (its type is a subtype of cloudify.nodes.Compute):
    • Install agent workers and required plugins on this host.
    • Execute cloudify.interfaces.monitoring_agent interface install and start operations. 1
  10. Execute cloudify.interfaces.lifecycle.poststart operation. 1
  11. Execute cloudify.interfaces.monitoring.start operation. 1
  12. Execute cloudify.interfaces.relationship_lifecycle.establish relationship operations.2

1. Execute the task mapped to the node’s lifecycle operation. (do nothing if no task is defined).
2. Execute all tasks mapped to this node’s relationship lifecycle operation. (Operations are executed in the order defined by the node template relationships)

The Uninstall Workflow

Workflow name: uninstall

Workflow description: Workflow for uninstalling applications.

Workflow parameters:

Workflow high-level pseudo-code:

For each node, for each node instance (in parallel):

  1. Wait for dependent node instances to be deleted. (Only start processing this node instance when the node instances dependent on it are deleted).
  2. Execute cloudify.interfaces.validation.delete operation. 1
  3. Execute cloudify.interfaces.monitoring.stop operation. 1
  4. Execute cloudify.interfaces.lifecycle.prestop operation. 1
  5. If node instance is host node (its type is a subtype of cloudify.nodes.Compute):
    • Execute cloudify.interfaces.monitoring_agent interface stop and uninstall operations. 1
    • Stop and uninstall agent workers.
  6. Execute cloudify.interfaces.lifecycle.stop operation.1
  7. Execute cloudify.interfaces.relationship_lifecycle.unlink relationship operations.2
  8. Execute cloudify.interfaces.lifecycle.delete operation.1
  9. Execute cloudify.interfaces.lifecycle.postdelete operation.1

1. Execute the task mapped to the node’s lifecycle operation. (do nothing if no task is defined).
2. Execute all tasks mapped to this node’s relationship lifecycle operation. (Operations are executed in the order defined by the node template relationships)

The Execute Operation Workflow

Workflow name: execute_operation

Workflow description: Generic workflow for executing arbitrary operations on nodes.

Workflow parameters:

Warning

Executing an operation on a node which has that interface operation but has not mapped it to any concrete implementation will simply do nothing. However, attempting to execute an operation on a node which doesn’t have the relevant interface operation will result in a workflow execution error. Use the filtering fields to ensure the operation is only executed on nodes which the operation might be relevant to.

Workflow high-level psuedo-code:

For each node, for each node instance:

  1. Filter out instances whose node is not in the node_ids list (unless its empty).
  2. Filter out instances whose id is not in the node_instance_ids list (unless its empty).
  3. Filter out instances whose node type is not in or a descendant of a type which is in the type_names list (unless its empty).

If run_by_dependency_order is set to true: create a task dependency between the following section’s (1) task for a given instance and the (3) task for all instances it depends on.1

For each of the remaining node instances:

  1. Send a node instance event about starting the execution operation.
  2. Execute the operation operation for the instance, with the operation_kwargs passed to the operation invocation.
  3. Send a node instance event about completing the execution of the operation.

1. Note that the dependency may be indirect, e.g. in a case where instance A is dependent on instance B, which is in turn dependent on instance C, and only B was filtered out, instance A’s operation execution will still only happen after instance C’s operation execution.

The Start Workflow

Workflow name: start

Workflow description: Can be used to start all, or a subset of, node templates.

This workflow is a wrapper for the execute_operation workflow, allowing the user to easily start the topology (or a subset thereof). Calling the start workflow is equivalent to calling execute_operation while passing cloudify.interfaces.lifecycle.start as the operation name.

Workflow parameters:

The Stop Workflow

Workflow name: stop

Workflow description: Can be used to stop all, or a subset of, node templates.

This workflow is a wrapper for the execute_operation workflow, allowing the user to easily stop the topology (or a subset thereof). Calling the stop workflow is equivalent to calling execute_operation while passing cloudify.interfaces.lifecycle.stop as the operation name.

Workflow parameters:

See workflow parameters for the start workflow above.

The Restart Workflow

Workflow name: restart

Workflow description: Can be used to restart all, or a subset of, node templates.

This workflow simply calls the stop workflow, followed by start.

Workflow parameters:

NOTE: The restart workflow performs all stop operations first, and then performs all start operations.

The Heal Workflow

Workflow name: heal

Workflow description: Heal node-instances of a deployment, or a subset of them. Run the check_status and heal interfaces for each selected instance, and in case those operations fail or are not declared, reinstall the node instance. Instances that pass their check_status call are not healed or reinstalled.

Workflow parameters:

Workflow high-level pseudo-code:

  1. Retrieve the node instances selected for healing.
    • if the node_instance_id parameter is empty, select all node instances in the deployment
    • if the node_instance_id parameter is specified, find the Compute node that the given instance is contained in, and select the whole subgraph - that Compute, and all instances contained in it
  2. If force_reinstall is set, reinstall all selected instances and exit.
  3. If check_status is set, run the cloudify.interfaces.validation.check_status operation for all selected instances. If the flag is not set, the result of the most recent check_status run will be used instead. An instance is considered healthy if the check_status operation returned a value, and not healthy if that operation raised an error. (Note: if an instance doesn’t declare a check_status at all, or if it was never run, the instance is considered NOT healthy).
  4. Compute the set of instances to be healed: those are instances which declare the cloudify.interfaces.lifecycle.heal interface, and their status is not healthy.
  5. Execute the heal operations on the instances to be healed. In dependency order, the following operations will be executed on each instance:
    • cloudify.interfaces.lifecycle.preheal
    • cloudify.interfaces.lifecycle.heal
    • cloudify.interfaces.lifecycle.postheal
  6. Compute the set of instances to be reinstalled:
    • instances that do not declare the cloudify.interfaces.lifecycle.heal operation
    • instanced that do declare the cloudify.interfaces.lifecycle.heal operation, but one of the operations (preheal, heal, postheal) raised an error
    • for every instance selected to be reinstalled, add the whole subgraph: all instances contained in them
  7. If allow_reinstall is set to false, and there are any instances to be reinstalled, throw an error and exit.
  8. Reinstall the selected instances:
    • run the uninstall operations on the selected instances, as in the uninstall workflow
    • run the install operations on the selected instances, as in the install workflow
    • establish all relationships between the reinstalled instances and other instances, by executing the preconfigure, postconfigure and establish operations, as in the install workflow

For example, if the blueprint defines these nodes: (passing_operation is an operation that always returns a value, and failing_operation is one that throws an error)

node_templates:
  webserver_host:  # connected to floating_ip
    type: cloudify.nodes.Compute
    interfaces:
      cloudify.interfaces.validation:
        check_status: failing_operation
      cloudify.interfaces.lifecycle:
        heal: passing_operation
    relationships:
      - target: floating_ip
        type: cloudify.relationships.connected_to

  webserver:  # contained in webserver_host
    type: cloudify.nodes.WebServer
    interfaces:
      cloudify.interfaces.validation:
        check_status: failing_operation
      cloudify.interfaces.lifecycle:
        heal: failing_operation
    relationships:
      - target: webserver_host
        type: cloudify.relationships.contained_in

  module:  # contained in webserver, connected to database
    type: cloudify.nodes.ApplicationModule
    relationships:
      - target: webserver
        type: cloudify.relationships.contained_in
      - target: database
        type: cloudify.relationships.connected_to

  database:
    type: cloudify.nodes.Database
    interfaces:
      cloudify.interfaces.validation:
        check_status: failing_operation
      cloudify.interfaces.lifecycle:
        heal: failing_operation

  floating_ip:
    type: cloudify.nodes.VirtualIP
    interfaces:
      cloudify.interfaces.validation:
        check_status: passing_operation

When the heal workflow is executed on a deployment created from this blueprint, without setting any parameters (ie. node_instance_id is not set, so all instances are healed):

  1. webserver_host fails check_status, so it is healed.
  2. webserver fails check_status and fails heal, so it is reinstalled. The whole subgraph needs to be reinstalled as well, therefore module is reinstalled (because it is contained in webserver)
  3. database fails its check_status and heal operations, therefore it will be reinstalled. Relationships to other instances (module) will be re-established.
  4. floating_ip passes its check_status call, so it is neither healed nor reinstalled. Relationships are NOT re-established.

The Scale Workflow

Workflow name: scale

Workflow description:

Scales out/in the node subgraph of the system topology applying the install/uninstall workflows’ logic respectively.

If the entity denoted by scalable_entity_name is a node template that is contained in a compute node (or is a compute node itself) and scale_compute is true, the node graph will consist of all nodes that are contained in the compute node which contains scalable_entity_name and the compute node itself. Otherwise, the subgraph will consist of all nodes that are contained in the node/scaling group denoted by scalable_entity_name.

In addition, nodes that are connected to nodes that are part of the contained subgraph will have their establish relationship operations executed during scale out and their unlink relationship operations executed during scale in.

Workflow parameters:

Workflow high-level pseudo-code:

  1. Retrieve the scaled node/scaling group, based on scalable_entity_name and scale_compute parameters.
  2. Start deployment modification, adding or removing node instances and relationship instances.
  3. If delta > 0:
    • Execute install lifecycle operations (create, configure, start) on added node instances.
    • Execute the establish relationship lifecycle operation for all affected relationships.
  4. If delta < 0:
    • Execute the unlink relationship lifecycle operation for all affected relationships.
    • Execute uninstall lifecycle operations (stop, delete) on removed node instances.

The Install New Agents Workflow

Workflow name: install_new_agents

Workflow description:

Installs agents on all VMs related to a particular deployment and connects them to the Cloudify Manager’s RabbitMQ instance. Please note that the old Manager has to be running during the execution of this workflow. What is worth mentioning as well is that the old agents don’t get uninstalled. This workflow’s common use case is executing it after having successfully restored a snapshot on a new Manager in order for the Manager to gain control over applications that have been orchestrated by the previous Manager.

Workflow parameters:

The Update workflow

Workflow name update

Workflow description:

Updates a deployment. See the deployment update page for details. This built-in workflow is usually not executed directly, but using the Deployment Updates API.

Workflow parameters:

Workflow high-level pseudo-code:

  1. Retrieve the new blueprint and the new inputs.
  2. Create a deployment plan (the same way as when first creating the deployment).
  3. Compute the differences between the existing deployment plan, and the newly-created plan.
  4. Based on the differences between the plans, generate and store deployment update steps.
  5. If preview is set, end the workflow.
  6. Set the updated attributes on the deployment itself, including:
    • labels
    • workflows
    • outputs
    • capabilities
    • description
    • inter-deployment dependencies
  7. Store newly-created nodes.
  8. Update changed nodes in storage, including setting new properties and relationships.
  9. Store newly-created node instances.
  10. Update node instance relationships in storage, adding newly-created relationships to existing node instances.
  11. If workflow_id is set, run that workflow.
  12. Otherwise, run the default update sequence:
    1. If skip_uninstall is not set, run the uninstall workflow on all removed node instances.
    2. If skip_install is not set, run the install workflow on all added node instances.
    3. Filter the node instances that need a configuration drift check: all instances in the started state, except the newly-created ones.
    4. If skip_drift_check is not set, run cloudify.interfaces.lifecycle.check_drift on those instances. Instances that don’t define this operation are considered drifted only if they changed in the blueprint (eg. a node property changed).
    5. Find the instances to be updated: instances that drifted, and define any of the update operations (see below). Also find the instances to be reinstalled: instances that drifted, and don’t define any of the update operations.
    6. If force_reinstall is set, all changed and drifted instances will be reinstalled, and no instances will be updated.
    7. If skip_heal is not set, run the heal workflow (with the check_status parameter enabled) on the instances to be updated. Instances that were unhealthy and couldn’t be healed, are no longer considered for update, but will need to be reinstalled.
    8. Run the update operations on all the instances to be updated, in dependency order:
      • cloudify.interfaces.lifecycle.update
      • cloudify.interfaces.lifecycle.update_config
      • cloudify.interfaces.lifecycle.update_apply
    9. Instances for which any of the update operations failed will be reinstalled.
    10. If skip_reinstall is not set, reinstall the instances to be reinstalled, by running the uninstall workflow followed by the uninstall workflow on them.
  13. Delete removed relationships, instances, and nodes, from storage.
  14. Update deployment execution schedules. New schedules are updated and changed schedules are updated, but deleted schedules are not removed from storage.
  15. Update stored operations with new inputs. This affects canceled executions, which can be resumed after the deployment update finishes, and use the new inputs.
  16. Finalize the deployment update.