Home >Backend Development >Golang >Deploying Serverless Applications on Google Cloud Run

Deploying Serverless Applications on Google Cloud Run

Susan Sarandon
Susan SarandonOriginal
2024-11-19 18:23:02545browse

Introduction

In this guide I will deploy the containers from Twitch's TeoMeWhy system, which is currently on AWS, and place it on GCP.

Current structure at AWS
Implantando Aplicações Serverless no Google Cloud Run

Architecture at GCP

Implantando Aplicações Serverless no Google Cloud Run

No complex automation tools will be used, everything will be done via the console, integrating with Github and deploying the images with each commit to main.
We will use:

  • Cloud Run - For web applications
  • Cloud SQL - For MySQL database
  • GCE - To run Teomebot
  • Cloud Storage - Object Storage (S3)
  • Cloud Build - To create application deployments
  • Secret Manager - Save application credentials securely.

Getting the applications

  1. Visit TeoMeWhy's GitHub and fork of applications related to the project.
  2. On the repository page, click Starred and then Fork. Implantando Aplicações Serverless no Google Cloud Run
  3. On the fork page, give the fork a name and click Create fork. Implantando Aplicações Serverless no Google Cloud Run
  4. Repeat the process for the other repositories in the project if you want to replicate the entire environment.
  5. Make the clone of the fork to adjust the application as needed before sending it to GCP.
git clone git@github.com:cslemes/points-to-go.git

Creating the Dockerfile to create the container image

Navigate to the cloned repository folder. The repository already contains a Dockerfile designed for Docker. Let's analyze it:

```
FROM golang:latest
WORKDIR /app/
COPY . .
RUN go build main.go
CMD ["./main"]
```

This Dockerfile is functional, but we will optimize it using multi-stage builds to reduce the size of the final image. Since Go doesn't require external dependencies, we can use a minimal base image like scratch.

  1. Swap the build image with a lighter version and name it for reference at other stages.

    FROM golang:1.23.1-alpine3.20 AS build
    
  2. Add the go mod download command to cache the dependencies and go mod verify to ensure they match the checksums in the go.sum.
    file

    RUN go mod download && go mod verify
    
  3. To optimize the binary for production, add the parameters below:

    • CGO_ENABLED=0: Disables CGO support. CGO is the functionality in Go that allows you to call C code, but when you disable it, you get a completely static binary, with no dependencies on external libraries.
    • GOARCH=amd64: Sets the target architecture for compilation. amd64 is the architecture used on machines with 64-bit processors, such as most modern servers and desktops.
    • GOOS=linux: Defines the target operating system for the compilation. Here it is configured for Linux, which means the generated binary will be executable on Linux systems.
    • go build -o /app/points: Compiles the code and saves the binary to the specified path (/app/points).
    • -a: Forces recompilation of all packages, including dependencies, even if they have not changed. It can be useful to ensure that everything is recompiled with the new flags and settings.
    • -ldflags="-s -w": These are size optimization flags.
    • -s removes the debug symbol table from the binary.
    • -w removes stack tracing information. These options reduce the size of the final binary.
    • -installsuffix cgo: Adds a suffix to the installation directory to differentiate binaries with CGO disabled. This can avoid conflicts with binaries that use CGO.
    git clone git@github.com:cslemes/points-to-go.git
    
  4. (optional) To make the image even smaller we can compress the binary using upx, upx (Ultimate Packer for eXecutables) is a tool that compresses executable binaries to reduce the size of files, such as Go binaries. The negative points are that There will be a longer build time, and a longer delay in container startup, so it must be evaluated which is most beneficial for its implementation. As the objective is to use it in Cloud Run which already has Cold Start, it pauses the container when it is not in use, we will not use it because it will increase the container's initialization time.

    ```
    FROM golang:latest
    WORKDIR /app/
    COPY . .
    RUN go build main.go
    CMD ["./main"]
    ```
    
  • upx --ultra-brute -qq points: Compresses the points binary aggressively, using all possible compression options and without displaying output messages.
  • upx -t points: Tests the compressed binary to ensure it still works correctly.
  • Now with the binary ready, let's do the second stage, which is copying the binary to a clean image.

    FROM golang:1.23.1-alpine3.20 AS build
    
    • scratch is a special base image in Docker that represents a completely empty image without any operating system or dependencies. Using FROM scratch is common in minimalist images for Go applications, where a static binary is sufficient.
  • Complete file

    git clone git@github.com:cslemes/points-to-go.git
    
  1. Comparing builds
  2. Dockerfile optimization and use of upx results in an image up to 3 times smaller.
  3. Analysis of the original image with Trivy shows 904 CVEs, while the scratch image is free of CVEs.
Imagem Tipo Tamanho
points-to-go Otimizado 14MB
points-to-go Com upx 5.77MB
points-to-go original 1.76GB

Implantando Aplicações Serverless no Google Cloud Run

  1. Repeat these settings for the other repositories.

Deploying the database

  1. In the console, access SQL in the side menu. Implantando Aplicações Serverless no Google Cloud Run
  2. Or search for "SQL" in the search bar and select SQL from the list. Implantando Aplicações Serverless no Google Cloud Run
  3. On the Cloud SQL page, click Create Instance and choose MySQL. Implantando Aplicações Serverless no Google Cloud Run
  4. Select Enterprise under editing.
  5. Under "Editing Presets", choose Sandbox.
  6. Leave the database version at MySQL 8.0. Implantando Aplicações Serverless no Google Cloud Run
  7. Give the instance a name under Instance ID and set a password.
  8. you can specify password policies by expanding password policy Implantando Aplicações Serverless no Google Cloud Run
  9. Define the zone and region, and we will leave it in a single zone. Implantando Aplicações Serverless no Google Cloud Run
  10. In customize instance we can adjust the hardware according to our needs, the options will vary depending on the edition we choose.
  11. Adjust CPU to 1 and Disk to 10GB as needed. Implantando Aplicações Serverless no Google Cloud Run
  12. In connections, uncheck Public IP and check Private IP.
  13. In private access connection click Configure Connection
  14. Select use an automatically allocated range
  15. Click Continue
  16. Click Create connection Implantando Aplicações Serverless no Google Cloud Run
  17. Click Create Instance and wait for it to be created. Implantando Aplicações Serverless no Google Cloud Run
  18. To connect to the instance, activate Cloud Shell and run:

    git clone git@github.com:cslemes/points-to-go.git
    

Implantando Aplicações Serverless no Google Cloud Run

  1. Enter the password assigned in the previous step.
  2. And then you will get a connection error, because the cloud shell does not have access to your private VPC. Implantando Aplicações Serverless no Google Cloud Run
  3. To simplify the connection via Cloud shell, let's edit the instance and mark public IP, in the appendix I will show how to create VPC peering to access the DB through Cloud Shell Implantando Aplicações Serverless no Google Cloud Run
  4. Try connecting again. Implantando Aplicações Serverless no Google Cloud Run
  5. Create the application database:

    ```
    FROM golang:latest
    WORKDIR /app/
    COPY . .
    RUN go build main.go
    CMD ["./main"]
    ```
    
  6. Cloud Shell still has a text editor based on VSCode, you can do some activities directly through it, it has 5GB of persistent volume in your /home , the hardware is an e2-small VM with 1vCPU and 1.7GB RAM.
    Implantando Aplicações Serverless no Google Cloud Run

Creating containers in Cloud Run

  1. In the Google Cloud console, access Cloud Run via the side menu or search bar. Implantando Aplicações Serverless no Google Cloud Run
  2. On the Cloud Run page, click Deploy Container and choose the Service option. Implantando Aplicações Serverless no Google Cloud Run
  3. Choosing the Deployment Method
  4. There are three options for deploying a service:
    • Use a container image from a registry.
    • Connect directly to a repository.
    • Create a function (using Cloud Functions, integrated with Run).
  5. For this guide, select Continuous Deployment with GitHub, allowing Google to configure the CI/CD pipeline automatically.
  6. Click Configure with Cloud Build.
    Implantando Aplicações Serverless no Google Cloud Run

  7. Configuring Connection to GitHub

  8. Click Authenticate to enable Google integration with your GitHub.

  9. Authorize access, choosing between allowing access to all repositories or just a specific one.

  10. After completing login, click Next.
    Implantando Aplicações Serverless no Google Cloud Run

  11. Choosing Build Type

  12. In the build step, select between using a Dockerfile or Supported Applications and Buildpacks from GCP.

  13. Opt for Dockerfile and adjust the path/filename if necessary
    Implantando Aplicações Serverless no Google Cloud Run

  14. Service Settings

  15. Configure the following parameters:

    • Authentication: Select Allow unauthenticated calls.
    • CPU Allocation: Choose CPU is allocated only during request processing.
    • Input Control: Select Internal. Implantando Aplicações Serverless no Google Cloud Run
  16. Adjust the container port as needed for the application.
    Implantando Aplicações Serverless no Google Cloud Run

  17. In the security tab, under service account, click create New service account
    Implantando Aplicações Serverless no Google Cloud Run

  18. Add roles

    • Storage Object Administrator
    • Cloud Run Administrator
    • Secret Manager secret advisor
    • Cloud SQL Client
    • Cloud Build Service Account
    • Artifact Registry Recorder
    • Service account user Implantando Aplicações Serverless no Google Cloud Run
  19. Analyzing the application code, we need to pass the environment variables to the database.

    git clone git@github.com:cslemes/points-to-go.git
    
  20. We will have to change the application code to the Cloud Sql standard that uses Unix sockets, and Cloud Run does not directly access the DB, it uses Cloud SQL Auth Proxy .

    ```
    FROM golang:latest
    WORKDIR /app/
    COPY . .
    RUN go build main.go
    CMD ["./main"]
    ```
    
  21. In the container, go to the variables and secrets tab and click on add variable.

  22. Add the necessary variables.

  23. The bank address follows the pattern, PROJECT_ID:REGION:INSTANCE_NAME, you can also get the name below, in item 16.
    Implantando Aplicações Serverless no Google Cloud Run

  24. The bank password we will put in REFERENCE A SECRET

  25. If the create new secret option is disabled, it is because you need to activate the Api.

  26. Click on CREATE NEW SECRET
    Implantando Aplicações Serverless no Google Cloud Run

  27. Define the secret name and click Create Secret
    Implantando Aplicações Serverless no Google Cloud Run

  28. In Cloud SQL Connections we will add the URL of the bank we created
    Implantando Aplicações Serverless no Google Cloud Run

  29. To create the application schema, the code requires us to pass the argument migrations=true.

  30. We will add migrations=true to the function argument, then remove it in the next container revision.
    Implantando Aplicações Serverless no Google Cloud Run

  31. Leave the other fields at default and click Create
    Implantando Aplicações Serverless no Google Cloud Run

  32. Testing the application using Thunder Client.

  33. creating a client
    Implantando Aplicações Serverless no Google Cloud Run

  34. reading registered customers
    Implantando Aplicações Serverless no Google Cloud Run

Creating Teomebot service

The chatbot will not be deployed on Cloud Run, it will be installed on Google Compute Engine (GCE). Unlike Cloud Run, Compute Engine is ideal because the chatbot needs to be continuously active to interact with the chat.

Additionally, we will cover the use of containers, secret management and Cloud Build configuration for deployment automation.

Creating a VM on Google Compute Engine (GCE)

  1. Access Compute Engine in the GCP Console side menu.
  2. Click Create Instance. Implantando Aplicações Serverless no Google Cloud Run
  3. Enter a name for the instance (e.g. teomebot-instance).
  4. Configure the region and zone as needed and write down this information for later use. Implantando Aplicações Serverless no Google Cloud Run
  5. In Machine Setup, choose E2 type, and in Preset, select e2-micro. Implantando Aplicações Serverless no Google Cloud Run
  6. Click on the Container tab and click on Deploy Container.
  7. Temporarily use the nginx:latest image to complete the initial configuration.
  8. Add the environment variables:
    • TWITCH_BOT: Bot name.
    • TWITCH_CHANNEL: Twitch channel name.
    • HOST_DB: Private IP of the Cloud SQL database.
    • USER_DB: Bank user.
    • PORT_DB: Bank port.
    • URL_POINTS: Service endpoint Points-to-Go.
  9. Click Select Implantando Aplicações Serverless no Google Cloud Run
  10. Under Identity and API Access, select the service account configured for this project. Implantando Aplicações Serverless no Google Cloud Run
  11. Leave the rest of the settings at default and click Create.

Adding Secrets with Secret Manager

  1. You may have noticed that we didn't pass the secrets, in the compute engine there is no simple way to pass them like in cloud run. My solution was to add a function to the application code to read information directly from the secret manager.
  2. Access Secret Manager in the GCP Console.
  3. Click Create Secret
  4. Enter a descriptive name (e.g. twitch-token) and add the corresponding value.
  5. Copy the path of the generated secret (e.g.: projects/123456/secrets/twitch-token/versions/latest). Implantando Aplicações Serverless no Google Cloud Run
    1. Create a new utils/gcp.go file
  6. Change utils/db.go to call the function, passing the secret manager path as a parameter.

    git clone git@github.com:cslemes/points-to-go.git
    
  7. Change main.go to get Twitch credentials

    ```
    FROM golang:latest
    WORKDIR /app/
    COPY . .
    RUN go build main.go
    CMD ["./main"]
    ```
    

    migration := flag.Bool("migrations", false, "Perform database migrations")

    flag.Parse()

    godotenv.Load()
    user := os.Getenv("TWITCH_BOT")
    // change
    oauth := utils.AccessSecretVersion("projects/551619572964/secrets/twitch-token/versions/latest")

    channel := os.Getenv("TWITCH_CHANNEL")
```

Cloud Build

Configuring Cloud Build

Cloud Build will be used to automate container image creation and deployment to Compute Engine.

  1. Create a cloudbuild.yaml file in the root of the repository with the content below:
  2. Substitutions

    FROM golang:1.23.1-alpine3.20 AS build
    

The _VERSION variable is set to a value that matches v1.0. with the hash of the current commit (${COMMIT_SHA}). This creates a unique version for each build, ensuring that each image is identifiable by version and commit.

  • Steps
    The steps section defines the steps that Cloud Build must perform. Here, we have four steps: build, push (twice) and update.

  • Step 1: Build the Docker Image

    RUN go mod download && go mod verify
    

This step runs a build of the Docker image:

  • "--no-cache": forces the build without using the cache.
  • "-t": defines tags for the created image:
    • gcr.io/$PROJECT_ID/teomebot:$_VERSION: image with the tag that uses the commit hash.
    • gcr.io/$PROJECT_ID/teomebot:latest: image with tag latest.
  • ".": defines the current directory as the build context.

The id: Build tag is an optional identifier for the step, useful for reference and debugging.

  • Step 2: Image Push with Version Tag
RUN CGO_ENABLED=0 GOARCH=amd64 GOOS=linux go build -o /app/points -a -ldflags="-s -w" -installsuffix cgo

This step pushes the image with the specific tag ($_VERSION) to the Google Container Registry, allowing the version generated in the build to be stored in the repository.

  • Step 3: Image Push with Tag latest

    RUN apk add --no-cache curl upx
    RUN upx --ultra-brute -qq points && upx -t points
    

This step pushes the image with the latest tag to the Google Container Registry, updating the latest image with the latest version.

  • Step 4: Container Update on a GCE Instance

     FROM scratch AS prod
     WORKDIR /app
     COPY --from=build /app/points /
     CMD ["./points"]
    

This step uses the gcloud command to update the container on a Google Compute Engine instance:

  • "teomebot-instance": specifies the name of the instance that runs the container.
  • --container-image: defines the container image that the instance should use. Here, use the latest version of the image.
  • --zone=$_DEPLOY_ZONE: uses a variable to specify the deployment zone.
  • --container-restart-policy=always: sets the container restart policy to always restart in case of failure.
  • Options

    git clone git@github.com:cslemes/points-to-go.git
    

The logging: CLOUD_LOGGING_ONLY option specifies that Cloud Build should only log to Cloud Logging, saving data and focus on GCP logs.

  • Final file

    ```
    FROM golang:latest
    WORKDIR /app/
    COPY . .
    RUN go build main.go
    CMD ["./main"]
    ```
    

Creating Trigger for Container Image Creation

Setting Up the Service Account

  1. Go to Cloud Build in the Google Cloud console.
  2. Go to Settings.
  3. Click on Service Account Permissions.
  4. Locate the service account created for Cloud Run.
  5. Activate the option Set as preferred service account
  6. Enable the Compute Instance Administrator role for the service account. Implantando Aplicações Serverless no Google Cloud Run Creating the Trigger
  7. In the side menu, click on Triggers and then on Create Trigger. Implantando Aplicações Serverless no Google Cloud Run
  8. Enter a descriptive name for the trigger.
  9. In Repositories, click Connect Repository and select the Teomebot repository. Implantando Aplicações Serverless no Google Cloud Run
  10. In Configuration, select the option Cloud Build Configuration File.
  11. Add the _DEPLOY_ZONE substitution variable with the value corresponding to the zone in which the instance was created.
  12. In service account, check the selected account if it is as configured in step 6. Implantando Aplicações Serverless no Google Cloud Run
  13. Click Save. Running the Trigger
  14. On the overview screen, in the newly created trigger line, click execute to run the process manually. Implantando Aplicações Serverless no Google Cloud Run
  15. In the process details, follow the image build steps to check for possible errors. Implantando Aplicações Serverless no Google Cloud Run

Testing the Application

  1. In the Compute Engine panel, copy the ssh command to access the instance, or use the ssh web client, and connect the instance.
  2. Connect to the instance and run the commands below to check the state of the container:

    git clone git@github.com:cslemes/points-to-go.git
    

Implantando Aplicações Serverless no Google Cloud Run

Resolving Certificate Problems

  1. If a certificate-related error occurs (caused by the scratch base image), replace it with the distroless image. In the Dockerfile, change the line that defines the base image from:

    git clone git@github.com:cslemes/points-to-go.git
    

to:

```
FROM golang:latest
WORKDIR /app/
COPY . .
RUN go build main.go
CMD ["./main"]
```

Dockerfile updated:

FROM golang:1.23.1-alpine3.20 AS build

Adjusting Permissions for Secret Manager

  1. Change service account scope to access Secret Manager.
  2. Go to the Google Cloud Console.
  3. In the side menu, go to Compute Engine > VM Instances.
  4. Find and click the name of your VM instance.
  5. On the VM details page, click Stop to shut down the instance (this step is necessary as the service account scope can only be modified when the instance is stopped).
  6. After the instance is stopped, click Edit at the top of the page.
  7. Scroll down to the Identity and Access API section.
  8. Under Service Account, select the service account that your application uses.
  9. Under API Access Scopes, select Allow Full Access to All Cloud APIs or click Set Specific API Access Scopes and add the scope https://www.googleapis.com/auth/cloud-platform.
  10. After adjusting the scope, click Save to apply the changes.
  11. Restart the instance by clicking Start.
  12. Or via the command line, stopping the instance, running the command and starting afterwards.

    RUN go mod download && go mod verify
    

Adding more containers

The other services follow the same points-to-go process, for services that communicate with each other, create environment variables to configure the endpoint addresses, which will always be https port 443.

For communication with other services, I adjusted the code to receive another environment variable with the service's url, in points for example it looked like this:

RUN CGO_ENABLED=0 GOARCH=amd64 GOOS=linux go build -o /app/points -a -ldflags="-s -w" -installsuffix cgo

Testing the bot

Testing the bot's communication with Twitch.

Implantando Aplicações Serverless no Google Cloud Run

Network Security Adjustment
After completing the tests, place the containers to be accessed only internally in the VPC.

Implantando Aplicações Serverless no Google Cloud Run

Conclusion

With this we have completed the migration of the TeoMeWhy system, the guide serves as a basis for migrating the other TeoMeWhy services.

The main objectives achieved were:

Technical Achievements

  • Migration of containerized applications to Cloud Run, allowing automatic scalability and cost reduction
  • Docker image optimization through multi-stage builds, significantly reducing image size and vulnerabilities
  • Implementation of a managed database with Cloud SQL, ensuring high availability and security
  • Automated CI/CD configuration using Cloud Build, enabling automatic deployments from GitHub
  • Secure credential management using Secret Manager

Architectural Improvements

  • Clear separation of responsibilities between services
  • Use of private connections for greater security
  • Implementation of serverless standards for cost optimization
  • Automation of build and deploy processes
  • Seamless integration with GitHub repositories

Benefits Obtained

  1. Costs: Cost reduction through the serverless model and resource optimization
  2. Maintainability: Ease of maintenance with automated deployments
  3. Security: Proper management of secrets and private connections
  4. Scalability: Ability to automatically scale according to demand
  5. Monitoring: Better infrastructure visibility through native GCP tools

Appendix

Enable the Secret Manager API

  1. In the Google Cloud console, search for Secret Manager API.
  2. Click on API in the search results. Implantando Aplicações Serverless no Google Cloud Run
  3. On the details screen, click Activate. Implantando Aplicações Serverless no Google Cloud Run

References

  • Github TeoMeWhy
  • Twitch Teo Me Why
  • Cloud Run Docs
  • Compute Engine Docs
  • Cloud Build Docs
  • Secret Manager Docs

The above is the detailed content of Deploying Serverless Applications on Google Cloud Run. For more information, please follow other related articles on the PHP Chinese website!

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn