DEV Community

Katie McLaughlin for Google Cloud

Posted on • Updated on

Migrating a Cloud Run deployment from shell to Terraform

There are many ways you can automate deployment of your infrastructure. It's perfectly valid to have a shell script that runs a series of create commands, but those commands themselves may not work too well when the entities they are trying to create already exist. So you could update that script to first check that the item already exists, and if not, create it. But then it won't work if you have any attribute updates you want to make. When you start including variables and preconditions into the mix, you end up with a script that is complicated and hard to debug.

This is where infrastructure automation comes in, and Infrastructure as Code (IaC). Using one of the many IaC tools available, you can make configurations that allow you to ensure your infrastructure is deployed reliably and cleanly, asserting configuration changes as required, in a way that is repeatable across your project.

In this post we'll show you how we can convert a shell script deployed Cloud Run service to a Terraform manifest, with all the considerations that involves.


The demo application

Today's demo application is going to be a base Hello World in Flask, which greets the value of 'TARGET', along with a Dockerfile to containerise it.

The original shell script

To start, here's a shell script we prepared earlier -- using gcloud, it configures and deploys a Cloud Run service to a project we've already set up with billing enabled, and the script works! ...mostly.

#!/bin/bash
export PROJECT_ID=glasnt-playground
gcloud config set project $PROJECT_ID

gcloud services enable run.googleapis.com

git clone https://gist.github.com/4db553e7f680784e8b910ca6de67c85b.git helloworld

cd helloworld

gcloud builds submit --tag gcr.io/${PROJECT_ID}/helloworld .

gcloud run deploy helloworld \
   --platform managed \
   --image gcr.io/${PROJECT_ID}/helloworld \
   --update-env-vars TARGET=gcloud \
   --allow-unauthenticated 
Enter fullscreen mode Exit fullscreen mode

Once deployed the service displays a hello message to the value of TARGET. We can get the service URL by describing the service, then using curl:

$ SERVICE_URL=$(gcloud run services describe helloworld --format "value(status.url)")
$ curl $SERVICE_URL
Hello gcloud!
Enter fullscreen mode Exit fullscreen mode

There are a few non-obvious problems with this script:

  • it only enables the Cloud Run API; this script also needs the Cloud Build API to be enabled, which is interactively asked for if it's not enabled when you run gcloud builds submit commands, thus breaking the automation by requiring human feedback.
  • it doesn't check if the Cloud Run API has already been enabled, wasting time asking for it to be re-enabled.
  • there are no checks if the service already exists, and it automatically forces the service to be public (which is a problem if it already exists and is private)
  • it doesn't define a region for the service, which if a region hasn't already been defined in the gcloud config, will cause the script to interactively ask for a region, which will break the automation loop by requiring human feedback.

We can work around some of these issues, and generally make our provisioning more robust, by converting this shell script into a Terraform manifest.

The rest of the post will explore that process. Some of the Terraform output assumes the original service already exists, so if you don't get the errors I get, then don't worry ✨

Setup Terraform

Before we begin, we need to install and configure Terraform. Follow the installation instructions for your platform (macOS users can brew install terraform). Check that it's all installed by checking the Terraform version:

$ terraform version
Terraform v0.12.24
Enter fullscreen mode Exit fullscreen mode

For authentication, it's recommended that we use a service account, so let's create one, then export a private key that Terraform can then use to act as this service account:

# Set the project ID
export PROJECT_ID=glasnt_playground
gcloud config set project $PROJECT_ID

# Create the service account
gcloud iam service-accounts create terraform \
    --display-name "Terraform Service Account"

# Grant role
gcloud projects add-iam-policy-binding ${PROJECT_ID} \
  --member serviceAccount:terraform@${PROJECT_ID}.iam.gserviceaccount.com \
  --role roles/owner

# create and save a local private key
gcloud iam service-accounts keys create ~/terraform-key.json \
  --iam-account terraform@${PROJECT_ID}.iam.gserviceaccount.com 

# store location of private key in environment that terraform can use
export GOOGLE_APPLICATION_CREDENTIALS=~/terraform-key.json

# enable the service that allows for automations
gcloud services enable cloudresourcemanager.googleapis.com
Enter fullscreen mode Exit fullscreen mode

Build the image

There are some things that Terraform can't do natively. One of those is building images.

So before we can define our service, we'll have to make sure we have an image ready. If you are following along and haven't got an image yet, you'd run:

$ git clone https://gist.github.com/4db553e7f680784e8b910ca6de67c85b.git helloworld

$ cd helloworld
$ gcloud builds submit --tag gcr.io/${PROJECT_ID}/helloworld
Enter fullscreen mode Exit fullscreen mode

Once this completes, we can confirm we have an image ready for use:

$ gcloud builds list --filter "images ~ hello" --sort-by "~create_time" --limit 1 \
 --format "table[no-heading](images[0])"
gcr.io/glasnt-playground/helloworld
Enter fullscreen mode Exit fullscreen mode

Create the manifest

We now need to define our Terraform manifest. Google Cloud resources are provided using the Google Cloud Terraform provider, where we'll define our Cloud Run service. The sample code for the resource types is very helpful to understand how to define resources.

In our example, we take the sample service, and substitute in the values from our original shell, including the explicit definition for the Terraform provider we need, and our environment variables.

Take a copy of this main.tf file into a new folder on your machine to follow along in the following steps. Make sure you edit the file to use your project ID!

Deploy the service

First time you run Terraform against a new manifest, you need to run the init command.

$ terraform init
Enter fullscreen mode Exit fullscreen mode

This will initialise Terraform for this particular manifest. Exactly what it does is output for us each time:

Initializing the backend...

Initializing provider plugins...
- Checking for available provider plugins...
- Downloading plugin for provider "google" (hashicorp/google) 3.16.0...

Terraform has been successfully initialized!

You may now begin working with Terraform. Try running "terraform plan" to see
any changes that are required for your infrastructure. All Terraform commands
should now work.

If you ever set or change modules or backend configuration for Terraform,
rerun this command to reinitialize your working directory. If you forget, other
commands will detect it and remind you to do so if necessary.
Enter fullscreen mode Exit fullscreen mode

The first time you run Terraform, this level of output may seem over the top, or even overwhelming. But take a moment to read what it's doing.

From here, we can plan what Terraform will do:

$ terraform plan
Refreshing Terraform state in-memory prior to plan...
The refreshed state will be used to calculate this plan, but will not be
persisted to local or remote state storage.

data.google_iam_policy.noauth: Refreshing state...

------------------------------------------------------------------------

An execution plan has been generated and is shown below.
Resource actions are indicated with the following symbols:
  + create

Terraform will perform the following actions:

  # google_cloud_run_service.helloworld will be created
  + resource "google_cloud_run_service" "helloworld" {

...

  # google_cloud_run_service_iam_policy.noauth will be created
  + resource "google_cloud_run_service_iam_policy" "noauth" {

...

Plan: 2 to add, 0 to change, 0 to destroy.

------------------------------------------------------------------------

Note: You didn't specify an "-out" parameter to save this plan, so Terraform
can't guarantee that exactly these actions will be performed if
"terraform apply" is subsequently run.


Enter fullscreen mode Exit fullscreen mode

This will show us exactly what Terraform plans to do. You should check to see what it expects to do. You can also see what properties will be made available after it exists.

We can now apply the manifest, as we tried to before:

$ terraform apply
...

Plan: 2 to add, 0 to change, 0 to destroy.

Do you want to perform these actions?
  Terraform will perform the actions described above.
  Only 'yes' will be accepted to approve.

  Enter a value:
Enter fullscreen mode Exit fullscreen mode

Terraform will only continue if we enter in exactly yes. The output before the confirmation will be the same as the terraform plan, so you don't need to re-run plan to see this output.

We can also skip this manual confirmation step by adding "-auto-approve" to our terraform apply command.

But for now, we should tell Terraform we're sure:

Do you want to perform these actions?
  Terraform will perform the actions described above.
  Only 'yes' will be accepted to approve.

  Enter a value: yes▊

google_cloud_run_service.helloworld: Creating...

Error: Error creating Service: googleapi: Error 409: Resource 'helloworld' already exists.

  on main.tf line 11, in resource "google_cloud_run_service" "helloworld":
  11: resource "google_cloud_run_service" "helloworld" {
Enter fullscreen mode Exit fullscreen mode

Uh oh.

Well, this is to be expected. We created our service with a shell script; Terraform wasn't to know it already existed until it tried to recreate it.

From here we have two options:

  • delete the service and get Terraform to re-create it
  • tell Terraform about the service, and then have it reconcile any differences.

The first method would be a case of running gcloud run service delete helloworld, but let's do it the smarter way.

When Terraform returns an Error 409: Resource already exists, it means that Terraform's local state doesn't match the project state. We can fix this by importing the state of our resource.

We want to make sure that Terraform is aware that the resource google_cloud_run_service helloworld maps to the real world Cloud Run service helloworld in region us-central1 under our project.

To do that, we can follow the hints from the import section of the documentation. Some gotchas here:

  • They define their only Cloud Run service as default; we named ours helloworld.
  • We both defined a location as being the region, us-central1.

Therefore, we can import our existing state by running:

$ terraform import google_cloud_run_service.helloworld us-central1/helloworld

google_cloud_run_service.helloworld: Importing from ID "us-central1/helloworld"...
google_cloud_run_service.helloworld: Import prepared!
  Prepared google_cloud_run_service for import
google_cloud_run_service.helloworld: Refreshing state... [id=locations/us-central1/namespaces/glasnt-playground/services/helloworld]

Import successful!

The resources that were imported are shown above. These resources are now in
your Terraform state and will henceforth be managed by Terraform.
Enter fullscreen mode Exit fullscreen mode

So now we can try that manifest again:

$ terraform apply

Plan: 1 to add, 1 to change, 0 to destroy.

Do you want to perform these actions?
  Terraform will perform the actions described above.
  Only 'yes' will be accepted to approve.

  Enter a value: yes

google_cloud_run_service.helloworld: Modifying... [id=locations/us-central1/namespaces/glasnt-playground/services/helloworld]
google_cloud_run_service.helloworld: Modifications complete after 6s [id=locations/us-central1/namespaces/glasnt-playground/services/helloworld]
google_cloud_run_service_iam_policy.noauth: Creating...
google_cloud_run_service_iam_policy.noauth: Creation complete after 4s [id=v1/projects/glasnt-playground/locations/us-central1/services/helloworld]

Apply complete! Resources: 1 added, 1 changed, 0 destroyed.
Enter fullscreen mode Exit fullscreen mode

We can check the service with curl:

$ curl $SERVICE_URL
Hello World!
Enter fullscreen mode Exit fullscreen mode

Hello... world? Ah, we didn't add a TARGET, so it no longer displays gcloud, so we know the service was re-deployed!

Deploy the service

Yes, we're doing this again.

Because one of the brilliant things about this setup is that you can re-apply the configurations to return the state (if changed), even if the items already exist.

You should be able to run terraform apply at any point, which will re-assert our configurations.

$ terraform apply 

...
Apply complete! Resources: 0 added, 0 changed, 0 destroyed.
Enter fullscreen mode Exit fullscreen mode

There are no modifications required, so there was no change to be made!

Update the service

In order to make changes, we need to change the manifest. Let's set a value for TARGET:

     spec {ye
       containers {
         image = "gcr.io/${local.project}/helloworld"
+        env {
+          name = "TARGET"
+          value = "terraform"
+        }
       }
     }
   }

Enter fullscreen mode Exit fullscreen mode

We can then run Terraform again and see the pending change:

$ terraform apply 

google_cloud_run_service.helloworld: Refreshing state... [id=locations/us-central1/namespaces/glasnt-playground/services/helloworld]

An execution plan has been generated and is shown below.
Resource actions are indicated with the following symbols:
  ~ update in-place

Terraform will perform the following actions:

...
          ~ spec {
                container_concurrency = 80

              ~ containers {
                    args    = []
                    command = []
                    image   = "gcr.io/glasnt-playground/helloworld"

                  + env {
                      + name  = "TARGET"
                      + value = "terraform"
                    }
...

Enter fullscreen mode Exit fullscreen mode

and apply the change:

  Enter a value: yes

google_cloud_run_service.helloworld: Modifying... [id=locations/us-central1/namespaces/glasnt-playground/services/helloworld]
google_cloud_run_service.helloworld: Still modifying... [id=locations/us-central1/namespaces/glasnt-playground/services/helloworld, 10s elapsed]
google_cloud_run_service.helloworld: Modifications complete after 19s [id=locations/us-central1/namespaces/glasnt-playground/services/helloworld]

Apply complete! Resources: 0 added, 1 changed, 0 destroyed.

Enter fullscreen mode Exit fullscreen mode

Yay! ✨

We can see that the service deployed successfully by checking the URL:

$ curl $SERVICE_URL
Hello terraform!
Enter fullscreen mode Exit fullscreen mode

Automatically deploying the service

One of the added benefits is that we can use this configuration in our deployment pipeline. Your project may not require this step, but it's always an option if that suits your deployment strategy.

Instead of manually running Terraform, we can replace the gcloud run deploy step of a standard Cloud Run deployment in Cloud Build configuration with a call to terraform. The configurations required for that are explained in the Terraform Cloud Builder documentation, along with using Google Storage as a Terraform backend.

Caveats

Terraform within Cloud Build is more complex, especially around the roles that need to be granted to the service account. In this example, helloworld was deployed with the default compute service account, which would not have been able to be edited without granting the Terraform service account the role/editor role, a permissions set too great for most applications.

Implementing Terraform within Cloud Build, and determining the minimum required permissions required, is outside the scope of this post.

Learn More

Oldest comments (1)

Collapse
 
mikevb3 profile image
Miguel Villarreal

Thank you Katie! i've been looking for resources of terraform and google cloud, and this setup is a great starting point!