Continuous Deployment with Architecture and Infrastructure Changes Using Cloudify

In our recent release of Cloudify 4.4, we made some really solid improvements and added great new features to make the user experience a breeze. Read about them here. One particular enhancement that is quite significant, especially for production deployments, is the Deployment Update feature, which is a built-in workflow that can be run from the UI or CLI.

Test-drive Cloudify’s Multi-Cloud Lab Free!    Get Lab

About Deployment Update

The deployment update feature enables users to add, remove, and connect new nodes in your existing deployment, while retaining the state of your current settings, instead of creating a new deployment.
A deployment can be updated in any of these three ways:

  • Upload a new blueprint and replace it with the current one
  • Change Inputs for the deployment, either with an inputs YAML file or providing each input value directly
  • Set specific Actions to be taken upon deployment update like enabling/disabling Install/Uninstall workflows on specific node instances

Updating Inputs in the UI, such as the web server URL or port endpoint, and setting specific Actions are simply a matter of changing the information and clicking “Update”. Those are trivial compared to migrating to a new service.
When an architectural change is required (such as in the example below), there is a lot more to consider. Therefore, we need to go back to the blueprint in order to make the necessary changes.
The update workflow execution occurs roughly in this order:

  • The unlink operation is executed in regard to each removed relationship.
  • The uninstall workflow is executed on each of the removed nodes.
  • Old plugins that were outdated are being uninstalled (if necessary).
  • The install workflow is executed on each of the added nodes.
  • The establish operation is executed in regard to each added relationship.
  • New plugins that were updated are being installed (if necessary).
  • The reinstall workflow (uninstall and install) is executed on each of the modified nodes and each of the nodes that were explicitly marked for reinstall.

The use case

In order to get a better understanding of what the Deployment Update does, let’s start with the assumption that you currently have a basic Continuous Integration pipeline. Github, Jenkins, a testing tool, artifact repo, and then Cloudify for deployment and lifecycle management of your infrastructure, application, network, etc. We will also take it as a given that you have deployed your application – in this example, an e-commerce site – with all the VMs, DBs, LBs, Firewall, and all that.
After doing some research, a decision has been made to migrate from the current local database running on AWS to Amazon RDS. But how can this be done in an automated way? After all, we will need to update infrastructure resources, application resources, networking resources.
In this example, you would be required to change infrastructure by moving from a VM running your application to the new service, create a new node to connect to the DBaaS ensuring zero downtime, decommission the old database nodes, update relationships to the existing nodes, and update inputs such as the URL and port.

How the magic happens

So, let’s see how it all goes down step-by-step in Cloudify.
First, we start with the below initial (sample) blueprint which will spin up the required VMs in AWS and installs the e-commerce app, DB, and FW. We also have our inputs file with the DB information – let’s call it inputs.yaml.

tosca_definitions_version: cloudify_dsl_1_3
imports:
  - http://www.getcloudify.org/spec/cloudify/4.4/types.yaml
  - http://www.getcloudify.org/spec/awssdk-plugin/2.4.1/plugin.yaml
inputs:
  db_url:
  db_port:
  db_password:
node_templates:
  host:
    type: cloudify.nodes.aws.ec2.Instance
  local_db:
    type: cloudify.nodes.DB
    properties:
      db_url: { get_input: db_url }
    relationships:
      - type: cloudify.relationships.contained_in
        target: host
  app:
    type: cloudify.nodes.WebApplication
    relationships:
      - type: cloudify.relationships.contained_in
        target: host
      - type: cloudify.relationships.connected_to
        target: local_db
  fw:
    type: cloudify.nodes.Firewall
    properties:
      rules:
        - port: { get_input: db_port }

app_local_db.yaml

db_url: http://10.0.20.34
db_port: 7876
db_password: "***********"

Initial inputs.yaml file

blueprint

Initial blueprint and input settings

Upload, Deployment, Install

Option 1: Using the Console UI, you can go to the Blueprints page and upload the blueprint, create a deployment (we’ll name it ecom_app), and execute the install workflow.
Option 2: From the terminal, run the cfy install command which will upload the blueprint to the manager, create the deployment, and run the install workflow: cfy install app_local_db.yaml -i inputs.yaml -n ecom-app

Cloudify console

ecom_app deployment

Deployment Update

Now that we have our running deployment, we will create the new blueprint (app_rds.yaml) with the changes we need to make and upload it to the manager. Use the link above to upload the blueprint or run cfy blueprints upload app_rds.yaml from the terminal.

tosca_definitions_version: cloudify_dsl_1_3
imports:
  - http://www.getcloudify.org/spec/cloudify/4.4/types.yaml
  - http://www.getcloudify.org/spec/awssdk-plugin/2.4.1/plugin.yaml
inputs:
  db_url:
  db_port:
  db_password:
node_templates:
  host:
    type: cloudify.nodes.aws.ec2.Instance
  dbaas:
    type: cloudify.nodes.aws.RDS
    properties:
      db_url: { get_input: db_url }
  app:
    type: cloudify.nodes.WebApplication
    relationships:
      - type: cloudify.relationships.contained_in
        target: host
      - type: cloudify.relationships.connected_to
        target: dbaas
  fw:
    type: cloudify.nodes.Firewall
    properties:
      rules:
        - port: { get_input: db_port }

app_rds.yaml

Note: For simplicity’s sake, we will show the Deployment Update from Cloudify Console only. To run the Deployment Update from the terminal, see the docs.
Go to the Deployments page, click the hamburger menu on the right-hand side of your deployment, and choose Update.

run deployment

New settings for deployment update

Once there, you can insert the ID of the blueprint being used for the deployment update and click on the Update button.
As you can see, we are updating a few things here:

  • Starting from the top, the new blueprint name, app_rds, tells Cloudify to check that blueprint
  • We also manually changed the db_port and db_url inputs
  • We ticked the Run install workflow first Action. This Action ensures that our new database is fully installed and configured before we remove the uninstall the node being changed, which means our application will effectively have zero downtime.

Deployment after update with tooltip notification

Deployment after update with tooltip notification

  • The deployment update is then started, and then the magic happens.
  • The new RDS service is installed and configured
  • The endpoint URL and port are updated
  • The firewall is updated with new rules to allow connectivity to the new RDS service
  • The new webapp relationship target is updated to connect to the RDS service
  • The old database is uninstalled

details of deployment

Details of deployment update

And that’s it. We just orchestrated the infrastructure migration of our VM-based database to AWS RDS using Cloudify!
As a bonus, we have a fantastic video detailing this feature and many others which you can watch below.

You rock, Tali!

comments

    Leave a Reply

    Your email address will not be published. Required fields are marked *

    Back to top