GSP662

Overview
There are many ways to deploy web sites within Google Cloud. Each solution offers different features, capabilities, and levels of control. Compute Engine offers a deep level of control over the infrastructure used to run a web site, but also requires a little more operational management compared to solutions like Google Kubernetes Engines (GKE), App Engine, or others. With Compute Engine, you have fine-grained control of aspects of the infrastructure, including the virtual machines, load balancers, and more.
In this lab you explore how to deploy a sample application, the "Fancy Store" e-commerce website, to show how a website can be deployed and scaled easily with Compute Engine.
What you'll learn
In this lab, you learn how to perform the following tasks:
By the end of the lab, you should have instances inside managed instance groups to provide autohealing, load balancing, autoscaling, and rolling updates for your website.
Setup and requirements
Before you click the Start Lab button
Read these instructions. Labs are timed and you cannot pause them. The timer, which starts when you click Start Lab, shows how long Google Cloud resources are made available to you.
This hands-on lab lets you do the lab activities in a real cloud environment, not in a simulation or demo environment. It does so by giving you new, temporary credentials you use to sign in and access Google Cloud for the duration of the lab.
To complete this lab, you need:
- Access to a standard internet browser (Chrome browser recommended).
Note: Use an Incognito (recommended) or private browser window to run this lab. This prevents conflicts between your personal account and the student account, which may cause extra charges incurred to your personal account.
- Time to complete the lab—remember, once you start, you cannot pause a lab.
Note: Use only the student account for this lab. If you use a different Google Cloud account, you may incur charges to that account.How to start your lab and sign in to the Google Cloud console
- 
Click the Start Lab button. If you need to pay for the lab, a dialog opens for you to select your payment method.
On the left is the Lab Details pane with the following: 
- The Open Google Cloud console button
- Time remaining
- The temporary credentials that you must use for this lab
- Other information, if needed, to step through this lab
 
- 
Click Open Google Cloud console (or right-click and select Open Link in Incognito Window if you are running the Chrome browser). The lab spins up resources, and then opens another tab that shows the Sign in page. Tip: Arrange the tabs in separate windows, side-by-side. Note: If you see the Choose an account dialog, click Use Another Account.
- 
If necessary, copy the Username below and paste it into the Sign in dialog. {{{user_0.username | "Username"}}}You can also find the Username in the Lab Details pane. 
- 
Click Next. 
- 
Copy the Password below and paste it into the Welcome dialog. {{{user_0.password | "Password"}}}You can also find the Password in the Lab Details pane. 
- 
Click Next. Important: You must use the credentials the lab provides you. Do not use your Google Cloud account credentials. 
 
Note: Using your own Google Cloud account for this lab may incur extra charges.
- 
Click through the subsequent pages: 
- Accept the terms and conditions.
- Do not add recovery options or two-factor authentication (because this is a temporary account).
- Do not sign up for free trials.
 
After a few moments, the Google Cloud console opens in this tab.
Note: To access Google Cloud products and services, click the Navigation menu or type the service or product name in the Search field.
 
Activate Cloud Shell
Cloud Shell is a virtual machine that is loaded with development tools. It offers a persistent 5GB home directory and runs on the Google Cloud. Cloud Shell provides command-line access to your Google Cloud resources.
- 
Click Activate Cloud Shell  at the top of the Google Cloud console. at the top of the Google Cloud console.
 
- 
Click through the following windows: 
- Continue through the Cloud Shell information window.
- Authorize Cloud Shell to use your credentials to make Google Cloud API calls.
 
When you are connected, you are already authenticated, and the project is set to your Project_ID, . The output contains a line that declares the Project_ID for this session:
Your Cloud Platform project in this session is set to {{{project_0.project_id | "PROJECT_ID"}}}
gcloud is the command-line tool for Google Cloud. It comes pre-installed on Cloud Shell and supports tab-completion.
- (Optional) You can list the active account name with this command:
gcloud auth list
- Click Authorize.
Output:
ACTIVE: *
ACCOUNT: {{{user_0.username | "ACCOUNT"}}}
To set the active account, run:
    $ gcloud config set account `ACCOUNT`
- (Optional) You can list the project ID with this command:
gcloud config list projectOutput:
[core]
project = {{{project_0.project_id | "PROJECT_ID"}}}
Note: For full documentation of gcloud, in Google Cloud, refer to the gcloud CLI overview guide.
Set your region and zone
Certain Compute Engine resources live in regions and zones. A region is a specific geographical location where you can run your resources. Each region has one or more zones.
Run the following gcloud commands in Cloud Shell to set the default region and zone for your lab:
gcloud config set compute/zone "{{{project_0.default_zone|ZONE}}}"
export ZONE=$(gcloud config get compute/zone)
gcloud config set compute/region "{{{project_0.default_region|REGION}}}"
export REGION=$(gcloud config get compute/region)
Task 1. Enable the Compute Engine API
gcloud services enable compute.googleapis.com
Task 2. Create a Cloud Storage bucket
You use a Cloud Storage bucket to house your built code as well as your startup scripts.
- From Cloud Shell, execute the following to create a new Cloud Storage bucket:
gsutil mb gs://fancy-store-{{{project_0.project_id | Project ID}}}Click Check my progress to verify the objective.
Create a Cloud Storage bucket
Task 3. Clone a source repository
This lab uses the existing Fancy Store e-commerce website based on the monolith-to-microservices repository as the basis for your website.
In this task, you clone the source code so you can focus on the aspects of deploying to Compute Engine. Later on in this lab, you perform a small update to the code to demonstrate the simplicity of updating on Compute Engine.
- Run the following commands to clone the source code and then navigate to the monolith-to-microservicesdirectory:
git clone https://github.com/googlecodelabs/monolith-to-microservices.git
cd ~/monolith-to-microservices
- Run the initial build of the code to allow the application to run locally with the following:
./setup.shIt takes a few minutes for this script to finish.
- Once completed, ensure Cloud Shell is running a compatible nodeJS version with the following command:
nvm install --lts
- Next, run the following to test the application, switch to the microservicesdirectory, and start the web server:
cd microservices
npm startYou should receive the following output.
Output:
Products microservice listening on port 8082!
Frontend microservice listening on port 8080!
Orders microservice listening on port 8081!
- Preview your application by clicking the web preview icon and selecting Preview on port 8080.

This opens a new window where you can see the frontend of Fancy Store.
Note: Within the Preview option, you should be able to see the Frontend; however, the Products and Orders functions do not work at this point, as those services are not yet exposed.
- Close this window after viewing the website and then press CTRL+C in the terminal window to stop the web server process.
Enable Gemini Code Assist in the Cloud Shell IDE
You can use Gemini Code Assist in an integrated development environment (IDE) such as Cloud Shell to receive guidance on code or solve problems with your code. Before you can start using Gemini Code Assist, you need to enable it.
- In Cloud Shell, enable the Gemini for Google Cloud API with the following command:
gcloud services enable cloudaicompanion.googleapis.com
- Click Open Editor on the Cloud Shell toolbar.
Note: To open the Cloud Shell Editor, click Open Editor on the Cloud Shell toolbar. You can switch between Cloud Shell and the code Editor by clicking Open Editor or Open Terminal, as required.
- 
In the left pane, click the Settings icon, then in the Settings view, search for Gemini Code Assist. 
- 
Locate and ensure that the checkbox is selected for Geminicodeassist: Enable, and close the Settings. 
- 
Click Cloud Code - No Project in the status bar at the bottom of the screen. 
- 
Authorize the plugin as instructed. If a project is not automatically selected, click Select a Google Cloud Project, and choose . 
- 
Verify that your Google Cloud project () displays in the Cloud Code status message in the status bar. 
Task 4. Create the Compute Engine instances
Now it's time to start deploying some Compute Engine instances!
In the sections that follow, you perform the following actions:
- Create a startup script to configure instances.
- Clone source code and upload to Cloud Storage.
- Deploy a Compute Engine instance to host the backend microservices.
- Reconfigure the frontend code to utilize the backend microservices instance.
- Deploy a Compute Engine instance to host the frontend microservice.
- Configure the network to allow communication.
Create the startup script
A startup script is used to instruct the instance what to do each time it is started. This way the instances are automatically configured.
- In the Cloud Shell terminal, run the following command to create a file called startup-script.sh:
touch ~/monolith-to-microservices/startup-script.sh
- Click Open Editor in the Cloud Shell toolbar to open the Code Editor.

- 
Navigate to the monolith-to-microservices folder. 
- 
Add the following code to the startup-script.shfile. Later on, you edit some of the code after it's added:
 
#!/bin/bash
# Install logging monitor. The monitor will automatically pick up logs sent to
# syslog.
curl -s "https://storage.googleapis.com/signals-agents/logging/google-fluentd-install.sh" | bash
service google-fluentd restart &
# Install dependencies from apt
apt-get update
apt-get install -yq ca-certificates git build-essential supervisor psmisc
# Install nodejs
mkdir /opt/nodejs
curl https://nodejs.org/dist/v16.14.0/node-v16.14.0-linux-x64.tar.gz | tar xvzf - -C /opt/nodejs --strip-components=1
ln -s /opt/nodejs/bin/node /usr/bin/node
ln -s /opt/nodejs/bin/npm /usr/bin/npm
# Get the application source code from the Google Cloud Storage bucket.
mkdir /fancy-store
gsutil -m cp -r gs://fancy-store-[DEVSHELL_PROJECT_ID]/monolith-to-microservices/microservices/* /fancy-store/
# Install app dependencies.
cd /fancy-store/
npm install
# Create a nodeapp user. The application will run as this user.
useradd -m -d /home/nodeapp nodeapp
chown -R nodeapp:nodeapp /opt/app
# Configure supervisor to run the node app.
cat >/etc/supervisor/conf.d/node-app.conf << EOF
[program:nodeapp]
directory=/fancy-store
command=npm start
autostart=true
autorestart=true
user=nodeapp
environment=HOME="/home/nodeapp",USER="nodeapp",NODE_ENV="production"
stdout_logfile=syslog
stderr_logfile=syslog
EOF
supervisorctl reread
supervisorctl update
- To update the startup-script.shfile, click the Gemini Code Assist: Smart Actions icon, and paste the following into the prompt to find and replace the text for icon, and paste the following into the prompt to find and replace the text for[DEVSHELL_PROJECT_ID].
As an application developer at Cymbal AI, update the "startup-script.sh" file. Replace [DEVSHELL_PROJECT_ID] with the project ID: {{{project_0.project_id | Project ID}}}.
- Press Enter to update the file. When prompted in the Gemini Diff view, click Accept.
The line of code within startup-script.sh should now resemble the following:
gs://fancy-store-{{{project_0.project_id | Project ID}}}/monolith-to-microservices/microservices/* /fancy-store/
- 
Click Save or press CTRL+S to save the startup-script.shfile, but do not close it yet.
 
- 
Refer to the bottom right of the Cloud Shell Code Editor and ensure "End of Line Sequence" is set to "LF", not "CRLF". 

- If this is set to CRLF, click CRLF and then select LF in the drop down.
- If this is already set to LF, leave it as is.
Use Gemini Code Assist to learn about the startup script file
To help you be more productive while minimizing context switching, Gemini Code Assist provides AI-powered smart actions directly in your code editor. For example, you can use the "Explain this" feature to let Gemini Code Assist give you more information about a particular file, block of code, or function.
In this section, you prompt Gemini Code Assist to provide more information about a startup script for a new team member who is unfamiliar with it.
- 
With the startup-script.shfile open, click the Gemini Code Assist: Smart Actions icon on the toolbar and select Explain this. icon on the toolbar and select Explain this.
 
- 
Gemini Code Assist opens a chat pane with the prefilled prompt of Explain this. In the inline text box of the Code Assist chat, replace the prefilled prompt with the following, and click Send:
 
You are an Application developer at Cymbal AI. A new team member is unfamiliar with this startup script. Explain this "startup-script.sh" file in detail, breaking down its key components used in the code. 
Do not suggest any improvements or changes to the file.The explanation for the code in the startup-script.sh file appears in the Gemini Code Assist chat.
- Close the startup-script.shfile.
Return to the Cloud Shell Terminal and run the following to copy the startup-script.sh file into your bucket:
gsutil cp ~/monolith-to-microservices/startup-script.sh gs://fancy-store-{{{project_0.project_id | Project ID}}}
It is now accessible at: https://storage.googleapis.com/[BUCKET_NAME]/startup-script.sh.
[BUCKET_NAME] represents the name of the Cloud Storage bucket. This is only viewable by authorized users and service accounts by default, and it is inaccessible in a web browser. Compute Engine instances are automatically able to access this through their service account.
The startup script performs the following tasks:
- Installs the Logging agent. The agent automatically collects logs from syslog.
- Installs Node.js and Supervisor; Supervisor runs the app as a daemon.
- Clones the app's source code from the Cloud Storage Bucket and installs dependencies.
- Configures Supervisor to run the app. Supervisor makes sure the app is restarted if it exits unexpectedly or is stopped by an admin or process. It also sends the app's stdout and stderr to syslog for the Logging agent to collect.
Copy code into the Cloud Storage bucket
When instances launch, they pull code from the Cloud Storage bucket, so you can store some configuration variables within the .env file of the code.
Note: You could also code this to pull environment variables from elsewhere. However, for demonstration purposes, this is a simple method to handle the configuration. In production, environment variables would likely be stored outside of the code.
- Run the following to copy the cloned code into your bucket:
cd ~
rm -rf monolith-to-microservices/*/node_modules
gsutil -m cp -r monolith-to-microservices gs://fancy-store-{{{project_0.project_id | Project ID}}}/
Note: Thenode_modules dependencies directories are deleted to ensure the copy is as fast and efficient as possible. These are recreated on the instances when they start up.
Click Check my progress to verify the objective.
Copy startup script and code to a Cloud Storage bucket
Deploy the backend instance
The first instance to be deployed is the backend instance, which houses the Orders and Products microservices.
Note: In a production environment, you may want to separate each microservice into its own instance and instance group to allow it to scale independently. For demonstration purposes, both backend microservices (Orders & Products) reside on the same instance and instance group.
- Execute the following command to create an e2-standard-2instance that is configured to use the startup script. It is tagged as abackendinstance so you can apply specific firewall rules to it later:
gcloud compute instances create backend \
    --zone={{{project_0.default_zone | zone}}} \
    --machine-type=e2-standard-2 \
    --tags=backend \
   --metadata=startup-script-url=https://storage.googleapis.com/fancy-store-{{{project_0.project_id | Project ID}}}/startup-script.shConfigure a connection to the backend
Before you deploy the frontend of the application, you need to update the configuration to point to the backend you just deployed.
- Retrieve the external IP address of the backend with the following command, look under the EXTERNAL_IPtab for the backend instance:
gcloud compute instances listExample output:
NAME: backend
ZONE: {{{project_0.default_zone | zone}}}
MACHINE_TYPE: e2-standard-2
PREEMPTIBLE: 
INTERNAL_IP: 10.142.0.2
EXTERNAL_IP: 35.237.245.193
STATUS: RUNNING
- 
Copy the External IP for the backend. 
- 
In the Cloud Shell Explorer, navigate to monolith-to-microservices>react-app.
 
- 
In the Code Editor, select View > Toggle Hidden Files in order to see the .envfile.
 
In the next step, you edit the .env file to point to the External IP of the backend. [BACKEND_ADDRESS] represents the External IP address of the backend instance determined from the recently executed gcloud command.
- In the .envfile, replacelocalhostwith your[BACKEND_ADDRESS]:
REACT_APP_ORDERS_URL=http://[BACKEND_ADDRESS]:8081/api/orders
REACT_APP_PRODUCTS_URL=http://[BACKEND_ADDRESS]:8082/api/products
- 
Save the file. 
- 
In Cloud Shell, run the following to rebuild react-app, which updates the frontend code:
 
cd ~/monolith-to-microservices/react-app
npm install && npm run-script build
- Then run the following to copy the application code into the Cloud Storage bucket:
cd ~
rm -rf monolith-to-microservices/*/node_modules
gsutil -m cp -r monolith-to-microservices gs://fancy-store-{{{project_0.project_id | Project ID}}}/Deploy the frontend instance
Now that the code is configured, you are ready to deploy the frontend instance.
- Execute the following to deploy the frontendinstance with a similar command as before. This instance is tagged asfrontendfor firewall purposes:
gcloud compute instances create frontend \
    --zone={{{project_0.default_zone | zone}}} \
    --machine-type=e2-standard-2 \
    --tags=frontend \
    --metadata=startup-script-url=https://storage.googleapis.com/fancy-store-{{{project_0.project_id | Project ID}}}/startup-script.sh
Note: The deployment command and startup script is used with both the frontend and backend instances for simplicity, and because the code is configured to launch all microservices by default. As a result, all microservices run on both the frontend and backend in this sample. In a production environment, you'd only run the microservices you need on each component.Configure the network
- Run the following to create firewall rules to allow access to port 8080 for the frontend, and ports 8081-8082 for the backend. These firewall commands use the tags assigned during instance creation for application:
gcloud compute firewall-rules create fw-fe \
    --allow tcp:8080 \
    --target-tags=frontend
gcloud compute firewall-rules create fw-be \
    --allow tcp:8081-8082 \
    --target-tags=backendThe website should now be fully functional.
- In order to navigate to the external IP of the frontend, you need to know the address. Run the following and look for the EXTERNAL_IP of thefrontendinstance:
gcloud compute instances listExample output:
NAME: backend
ZONE: us-central1-f
MACHINE_TYPE: e2-standard-2
PREEMPTIBLE:
INTERNAL_IP: 10.128.0.2
EXTERNAL_IP: 34.27.178.79
STATUS: RUNNING
NAME: frontend
ZONE: us-central1-f
MACHINE_TYPE: e2-standard-2
PREEMPTIBLE:
INTERNAL_IP: 10.128.0.3
EXTERNAL_IP: 34.172.241.242
STATUS: RUNNING
It may take several minutes for the instance to start and be configured.
- 
Wait 3 minutes and then open a new browser tab and browse to http://[FRONTEND_ADDRESS]:8080to access the website, where [FRONTEND_ADDRESS] is the frontend EXTERNAL_IP determined prior.
 
- 
Try navigating to the Products and Orders pages; these should now work. 

Click Check my progress to verify the objective.
Deploy instances and configure the network
Task 5. Create managed instance groups
To allow the application to scale, managed instance groups are created and use the frontend and backend instances as Instance Templates.
A managed instance group (MIG) contains identical instances that you can manage as a single entity in a single zone. Managed instance groups maintain high availability of your apps by proactively keeping your instances available, that is, in the RUNNING state. You intend using managed instance groups for your frontend and backend instances to provide autohealing, load balancing, autoscaling, and rolling updates.
Create an instance template from a source instance
Before you can create a managed instance group, you have to first create an instance template to be the foundation for the group. Instance templates allow you to define the machine type, boot disk image or container image, network, and other instance properties to use when creating new VM instances. You can use instance templates to create instances in a managed instance group or even to create individual instances.
To create the instance template, use the existing instances you created previously.
- First, run the following to stop both instances:
gcloud compute instances stop frontend --zone={{{project_0.default_zone | zone}}}
gcloud compute instances stop backend --zone={{{project_0.default_zone | zone}}}
- Then, create the instance template from each of the source instances with the following commands:
gcloud compute instance-templates create fancy-fe \
    --source-instance-zone={{{project_0.default_zone | zone}}} \
    --source-instance=frontend
gcloud compute instance-templates create fancy-be \
    --source-instance-zone={{{project_0.default_zone | zone}}} \
    --source-instance=backend
- Run the following to confirm the instance templates were created:
gcloud compute instance-templates listExample output:
NAME: fancy-be
MACHINE_TYPE: e2-standard-2
PREEMPTIBLE: 
CREATION_TIMESTAMP: 2023-07-25T14:52:21.933-07:00
NAME: fancy-fe
MACHINE_TYPE: e2-standard-2
PREEMPTIBLE: 
CREATION_TIMESTAMP: 2023-07-25T14:52:15.442-07:00
- With the instance templates created, run the following to delete the backendVM to save resource space:
gcloud compute instances delete backend --zone={{{project_0.default_zone | zone}}}
- Type and enter y when prompted.
Normally, you could delete the frontend VM as well, but you need to use it to update the instance template later in the lab.
Create managed instance groups
- Next, run the following commands to create two managed instance groups, one for the frontend and one for the backend:
gcloud compute instance-groups managed create fancy-fe-mig \
    --zone={{{project_0.default_zone | zone}}} \
    --base-instance-name fancy-fe \
    --size 2 \
    --template fancy-fe
gcloud compute instance-groups managed create fancy-be-mig \
    --zone={{{project_0.default_zone | zone}}} \
    --base-instance-name fancy-be \
    --size 2 \
    --template fancy-beThese managed instance groups use the instance templates and are configured for two instances each within each group to start. The instances are automatically named based on the base-instance-name specified with random characters appended.
- Run the following to ensure that for your application, the frontendmicroservice runs on port 8080, and thebackendmicroservice runs on port 8081 fororders, and port 8082 forproducts:
gcloud compute instance-groups set-named-ports fancy-fe-mig \
    --zone={{{project_0.default_zone | zone}}} \
    --named-ports frontend:8080
gcloud compute instance-groups set-named-ports fancy-be-mig \
    --zone={{{project_0.default_zone | zone}}} \
    --named-ports orders:8081,products:8082Since these are non-standard ports, you specify named ports to identify these. Named ports are key:value pair metadata representing the service name and the port that it's running on. Named ports can be assigned to an instance group, which indicates that the service is available on all instances in the group. This information is used by the HTTP Load Balancing service that you configure later.
Configure autohealing
To improve the availability of the application itself and to verify it is responding, configure an autohealing policy for the managed instance groups.
An autohealing policy relies on an application-based health check to verify that an app is responding as expected. Checking that an app responds is more precise than simply verifying that an instance is in a RUNNING state, which is the default behavior.
Note: Separate health checks are used for load balancing and for autohealing. Health checks for load balancing can and should be more aggressive because these health checks determine whether an instance receives user traffic. You want to catch non-responsive instances quickly so you can redirect traffic if necessary.
In contrast, health checking for autohealing causes Compute Engine to proactively replace failing instances, so this health check should be more conservative than a load balancing health check.
- Run the following to create a health check that repairs the instance if it returns "unhealthy" 3 consecutive times for the frontendandbackend:
gcloud compute health-checks create http fancy-fe-hc \
    --port 8080 \
    --check-interval 30s \
    --healthy-threshold 1 \
    --timeout 10s \
    --unhealthy-threshold 3
gcloud compute health-checks create http fancy-be-hc \
    --port 8081 \
    --request-path=/api/orders \
    --check-interval 30s \
    --healthy-threshold 1 \
    --timeout 10s \
    --unhealthy-threshold 3
- Run the following to create a firewall rule that allows the health check probes to connect to the microservices on ports 8080-8081:
gcloud compute firewall-rules create allow-health-check \
    --allow tcp:8080-8081 \
    --source-ranges 130.211.0.0/22,35.191.0.0/16 \
    --network default
- Apply the health checks to their respective services with the following commands:
gcloud compute instance-groups managed update fancy-fe-mig \
    --zone={{{project_0.default_zone | zone}}} \
    --health-check fancy-fe-hc \
    --initial-delay 300
gcloud compute instance-groups managed update fancy-be-mig \
    --zone={{{project_0.default_zone | zone}}} \
    --health-check fancy-be-hc \
    --initial-delay 300
Note: It can take 15 minutes before autohealing begins monitoring instances in the group.
- Continue with the lab to allow some time for autohealing to monitor the instances in the group. You intend simulating a failure to test the autohealing at the end of the lab.
Click Check my progress to verify the objective.
Create managed instance groups
Task 6. Create load balancers
To complement your managed instance groups, use HTTP(S) Load Balancers to serve traffic to the frontend and backend microservices, and use mappings to send traffic to the proper backend services based on pathing rules. This exposes a single load balanced IP for all services.
You can learn more about the Load Balancing options on Google Cloud: Overview of Load Balancing.
Create the HTTP(S) load balancer
Google Cloud offers many different types of load balancers. For this lab, you use an HTTP(S) Load Balancer for your traffic. An HTTP load balancer is structured as follows:
- A forwarding rule directs incoming requests to a target HTTP proxy.
- The target HTTP proxy checks each request against a URL map to determine the appropriate backend service for the request.
- The backend service directs each request to an appropriate backend based on serving capacity, zone, and instance health of its attached backends. The health of each backend instance is verified using an HTTP health check. If the backend service is configured to use an HTTPS or HTTP/2 health check, the request is encrypted on its way to the backend instance.
- Sessions between the load balancer and the instance can use the HTTP, HTTPS, or HTTP/2 protocol. If you use HTTPS or HTTP/2, each instance in the backend services must have an SSL certificate.
Note: For demonstration purposes in order to avoid SSL certificate complexity, you use HTTP instead of HTTPS. For production, it is recommended to use HTTPS for encryption wherever possible.
- Run the following to create health checks that are used to determine which instances are capable of serving traffic for each service:
gcloud compute http-health-checks create fancy-fe-frontend-hc \
  --request-path / \
  --port 8080
gcloud compute http-health-checks create fancy-be-orders-hc \
  --request-path /api/orders \
  --port 8081
gcloud compute http-health-checks create fancy-be-products-hc \
  --request-path /api/products \
  --port 8082
Note: These health checks are for the load balancer and only handle directing traffic from the load balancer; they do not cause the managed instance groups to recreate instances.
- Run the following to create backend services that are the target for load-balanced traffic; the backend services use the health checks and named ports you created:
gcloud compute backend-services create fancy-fe-frontend \
  --http-health-checks fancy-fe-frontend-hc \
  --port-name frontend \
  --global
gcloud compute backend-services create fancy-be-orders \
  --http-health-checks fancy-be-orders-hc \
  --port-name orders \
  --global
gcloud compute backend-services create fancy-be-products \
  --http-health-checks fancy-be-products-hc \
  --port-name products \
  --global
- Run the following to add the load balancer's backend services:
gcloud compute backend-services add-backend fancy-fe-frontend \
  --instance-group-zone={{{project_0.default_zone | zone}}} \
  --instance-group fancy-fe-mig \
  --global
gcloud compute backend-services add-backend fancy-be-orders \
  --instance-group-zone={{{project_0.default_zone | zone}}} \
  --instance-group fancy-be-mig \
  --global
gcloud compute backend-services add-backend fancy-be-products \
  --instance-group-zone={{{project_0.default_zone | zone}}} \
  --instance-group fancy-be-mig \
  --global
- Run the following command to create a URL map that defines which URLs are directed to which backend services:
gcloud compute url-maps create fancy-map \
  --default-service fancy-fe-frontend
- Run the following command to create a path matcher that allows the /api/ordersand/api/productspaths to route to their respective services:
gcloud compute url-maps add-path-matcher fancy-map \
   --default-service fancy-fe-frontend \
   --path-matcher-name orders \
   --path-rules "/api/orders=fancy-be-orders,/api/products=fancy-be-products"
- Run the following to create the proxy, which ties to the URL map:
gcloud compute target-http-proxies create fancy-proxy \
  --url-map fancy-map
- Run the following command to create a global forwarding rule that ties a public IP address and port to the proxy:
gcloud compute forwarding-rules create fancy-http-rule \
  --global \
  --target-http-proxy fancy-proxy \
  --ports 80Click Check my progress to verify the objective.
Create HTTP(S) load balancers
Update the configuration
Now that you have a new static IP address, update the code on the frontend to point to this new address instead of the ephemeral address used earlier that pointed to the backend instance.
- In Cloud Shell, run the following to change to the react-appfolder that houses the.envfile, which holds the configuration:
cd ~/monolith-to-microservices/react-app/
- Find the IP address for the Load Balancer by running the following command:
gcloud compute forwarding-rules list --globalExample output:
NAME: fancy-http-rule
REGION:
IP_ADDRESS: 34.111.203.235
IP_PROTOCOL: TCP
TARGET: fancy-proxy
- Return to the Cloud Shell Editor and edit the .envfile again to point to Public IP of Load Balancer. [LB_IP] represents the External IP address of the backend instance determined previously.
REACT_APP_ORDERS_URL=http://[LB_IP]/api/orders
REACT_APP_PRODUCTS_URL=http://[LB_IP]/api/products
Note: The ports are removed in the new address because the load balancer is configured to handle this forwarding for you.
- 
Save the file. 
- 
Run the following commands to rebuild react-app, which updates the frontend code:
 
cd ~/monolith-to-microservices/react-app
npm install && npm run-script build
- Run the following to copy the application code into your bucket:
cd ~
rm -rf monolith-to-microservices/*/node_modules
gsutil -m cp -r monolith-to-microservices gs://fancy-store-{{{project_0.project_id | Project ID}}}/Update the frontend instances
Now that there is new code and a configuration, you want the frontend instances within the managed instance group to pull the new code.
- Since your instances pull the code at startup, run the following command to issue a rolling restart:
gcloud compute instance-groups managed rolling-action replace fancy-fe-mig \
    --zone={{{project_0.default_zone | zone}}} \
    --max-unavailable 100%
Note: In this example of a rolling replace, you specifically state that all machines can be replaced immediately through the--max-unavailable parameter. Without this parameter, the command would keep an instance alive while restarting others to ensure availability. For testing purposes, you specify to replace all immediately for speed.
Click Check my progress to verify the objective.
Update the frontend instances
Test the website
- Wait three minutes after issuing the rolling-action replacecommand in order to give the instances time to be processed, and then check the status of the managed instance group. Run the following to confirm the service is listed as HEALTHY:
watch -n 2 gcloud compute backend-services get-health fancy-fe-frontend --global
- Wait until the two services are listed as HEALTHY.
Example output:
backend: https://www.googleapis.com/compute/v1/projects/my-gce-codelab/zones/us-central1-a/instanceGroups/fancy-fe-mig
status:
healthStatus:
- healthState: HEALTHY
  instance: https://www.googleapis.com/compute/v1/projects/my-gce-codelab/zones/us-central1-a/instances/fancy-fe-x151
  ipAddress: 10.128.0.7
  port: 8080
- healthState: HEALTHY
  instance: https://www.googleapis.com/compute/v1/projects/my-gce-codelab/zones/us-central1-a/instances/fancy-fe-cgrt
  ipAddress: 10.128.0.11
  port: 8080
  kind: compute#backendServiceGroupHealth
  
Note: If one instance encounters an issue and is UNHEALTHY it should automatically be repaired. Wait for this to happen. 
If neither instance enters a HEALTHY state after waiting a little while, something is wrong with the setup of the frontend instances that accessing them on port 8080 doesn't work. Test this by browsing to the instances directly on port 8080.
- Once both items appear as HEALTHY on the list, exit the watchcommand by pressing CTRL+C.
Note: The application is accessible via http://[LB_IP] where [LB_IP] is the IP_ADDRESS specified for the Load Balancer, which can be found with the following command:gcloud compute forwarding-rules list --global
You check the application later in the lab.
Task 7. Scale Compute Engine
So far, you have created two managed instance groups with two instances each. This configuration is fully functional, but a static configuration regardless of load. Next, you create an autoscaling policy based on utilization to automatically scale each managed instance group.
Automatically resize by utilization
- To create the autoscaling policy, execute the following:
gcloud compute instance-groups managed set-autoscaling \
  fancy-fe-mig \
  --zone={{{project_0.default_zone | zone}}} \
  --max-num-replicas 2 \
  --target-load-balancing-utilization 0.60
gcloud compute instance-groups managed set-autoscaling \
  fancy-be-mig \
  --zone={{{project_0.default_zone | zone}}} \
  --max-num-replicas 2 \
  --target-load-balancing-utilization 0.60These commands create an autoscaler on the managed instance groups that automatically adds instances when utilization is above 60% utilization, and removes instances when the load balancer is below 60% utilization.
Enable the content delivery network
Another feature that can help with scaling is to enable a Content Delivery Network service, to provide caching for the frontend.
- Execute the following command on the frontend service:
gcloud compute backend-services update fancy-fe-frontend \
    --enable-cdn --globalWhen a user requests content from the HTTP(S) load balancer, the request arrives at a Google Front End (GFE), which first looks in the Cloud CDN cache for a response to the user's request. If the GFE finds a cached response, the GFE sends the cached response to the user. This is called a cache hit.
If the GFE can't find a cached response for the request, the GFE makes a request directly to the backend. If the response to this request is cacheable, the GFE stores the response in the Cloud CDN cache so that the cache can be used for subsequent requests.
Click Check my progress to verify the objective.
Scale Compute Engine
Task 8. Update the website
Update the instance template
Existing instance templates are not editable; however, since your instances are stateless and all configuration is done through the startup script, you only need to change the instance template if you want to change the template settings. In this task, you make a simple change to use a larger machine type and push that out.
Complete the steps that follow to perform the following actions:
- 
Update the frontendinstance, which acts as the basis for the instance template. During the update, put a file on the updated version of the instance template's image, then update the instance template, roll out the new template, and then confirm the file exists on the managed instance group instances.
 
- 
Modify the machine type of your instance template by switching from the e2-standard-2machine type toe2-small.
 
- Run the following command to modify the machine type of the frontend instance:
gcloud compute instances set-machine-type frontend \
  --zone={{{project_0.default_zone | zone}}} \
  --machine-type e2-small
- Run the following command to create the new Instance Template:
gcloud compute instance-templates create fancy-fe-new \
    --region=$REGION \
    --source-instance=frontend \
    --source-instance-zone={{{project_0.default_zone | zone}}}
- Roll out the updated instance template to the Managed Instance Group with the following command:
gcloud compute instance-groups managed rolling-action start-update fancy-fe-mig \
  --zone={{{project_0.default_zone | zone}}} \
  --version template=fancy-fe-new
- Wait 3 minutes, and then run the following to monitor the status of the update:
watch -n 2 gcloud compute instance-groups managed list-instances fancy-fe-mig \
  --zone={{{project_0.default_zone | zone}}}This process takes several minutes.
Ensure you have at least one instance in the following condition before proceeding:
- STATUS: RUNNING
- ACTION set to None
- INSTANCE_TEMPLATE: the new template name (fancy-fe-new)
- 
Copy the name of one of the machines listed for use in the next command. 
- 
Press CTRL+C to exit the watchprocess.
 
- 
Run the following to see if the virtual machine is using the new machine type (e2-small), where [VM_NAME] is the newly created instance: 
gcloud compute instances describe [VM_NAME] --zone={{{project_0.default_zone | zone}}} | grep machineTypeExpected example output:
machineType: https://www.googleapis.com/compute/v1/projects/project-name/zones/us-central1-f/machineTypes/e2-small
Make changes to the website
Scenario: Your marketing team has asked you to change the homepage for your site. They think it should be more representative of who the company is and what you actually sell.
In this section, you add some text to the homepage to make the marketing team happy! It looks like one of the developers has already created the changes with the file name index.js.new. You can just copy this file to index.js and the changes should be reflected. Perform the instructions that follow to make the appropriate changes.
- Run the following commands to copy the updated file to the correct file name:
cd ~/monolith-to-microservices/react-app/src/pages/Home
mv index.js.new index.js
- Print the file contents to verify the changes:
cat ~/monolith-to-microservices/react-app/src/pages/Home/index.jsThe resulting code should resemble the following output.
Output:
/*
Copyright 2019 Google LLC
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
    https://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
import React from "react";
import { Box, Paper, Typography } from "@mui/material";
export default function Home() {
  return (
    <Box sx={{ flexGrow: 1 }}>
      <Paper
        elevation={3}
        sx={{
          width: "800px",
          margin: "0 auto",
          padding: (theme) => theme.spacing(3, 2),
        }}
      >
        <Typography variant="h5">Welcome to the Fancy Store!</Typography>
        <br />
        <Typography variant="body1">
          Take a look at our wide variety of products.
        </Typography>
      </Paper>
    </Box>
  );
}
You updated the React components, but you need to build the React app to generate the static files.
- Run the following command to build the React app and copy it into the monolith public directory:
cd ~/monolith-to-microservices/react-app
npm install && npm run-script build
- Then run the following to re-push this code to the bucket:
cd ~
rm -rf monolith-to-microservices/*/node_modules
gsutil -m cp -r monolith-to-microservices gs://fancy-store-{{{project_0.project_id | Project ID}}}/Push changes with rolling replacements
- Now run the following to force all instances to be replaced to pull the update:
gcloud compute instance-groups managed rolling-action replace fancy-fe-mig \
  --zone={{{project_0.default_zone | zone}}} \
  --max-unavailable=100%Note: In this example of a rolling replace, you specifically state that all machines can be replaced immediately through the --max-unavailable parameter. Without this parameter, the command would keep an instance alive while replacing others. For testing purposes, you specify to replace all immediately for speed. In production, leaving a buffer would allow the website to continue serving the website while updating.
Click Check my progress to verify the objective.
Update the website
- Wait three minutes after issuing the rolling-action replacecommand in order to give the instances time to be processed, and then check the status of the managed instance group. Run the following to confirm the service is listed as HEALTHY:
watch -n 2 gcloud compute backend-services get-health fancy-fe-frontend --global
- Wait a few moments for both services to appear and become HEALTHY.
Example output:
backend: https://www.googleapis.com/compute/v1/projects/my-gce-codelab/zones/us-central1-a/instanceGroups/fancy-fe-mig
status:
healthStatus:
- healthState: HEALTHY
  instance: https://www.googleapis.com/compute/v1/projects/my-gce-codelab/zones/us-central1-a/instances/fancy-fe-x151
  ipAddress: 10.128.0.7
  port: 8080
- healthState: HEALTHY
  instance: https://www.googleapis.com/compute/v1/projects/my-gce-codelab/zones/us-central1-a/instances/fancy-fe-cgrt
  ipAddress: 10.128.0.11
  port: 8080
  kind: compute#backendServiceGroupHealth
  
- 
Once items appear in the list with a HEALTHY status, exit the watchcommand by pressing CTRL+C.
 
- 
Browse to the website via http://[LB_IP]where [LB_IP] is the IP_ADDRESS specified for the Load Balancer, which can be found with the following command:
 
gcloud compute forwarding-rules list --globalThe new website changes should now be visible.
Simulate failure
In order to confirm that the health check works, you decide to log into an instance and stop the services.
- To find an instance name, execute the following:
gcloud compute instance-groups list-instances fancy-fe-mig --zone={{{project_0.default_zone | zone}}}
- Copy an instance name, then run the following to SSH into the instance, where INSTANCE_NAME is one of the instances from the list:
gcloud compute ssh [INSTANCE_NAME] --zone={{{project_0.default_zone | zone}}}
- 
Type in "y" to confirm, and press Enter twice to not use a password. 
- 
Within the instance, run the following to use supervisorctlto stop the application:
 
sudo supervisorctl stop nodeapp; sudo killall node
- Run the following command to exit the instance:
exit
- Monitor the repair operations:
watch -n 2 gcloud compute operations list \
--filter='operationType~compute.instances.repair.*'This process takes several minutes to complete.
Look for the following example output.
Output:
NAME: repair-1755080598062-63c3c8b99843b-eed8dabc-f1833ea3
TYPE: compute.instances.repair.recreateInstance
TARGET: us-east4-c/instances/fancy-fe-tn40
HTTP_STATUS: 200
STATUS: DONE
TIMESTAMP: 2025-08-13T03:23:18.062-07:00
The managed instance group recreated the instance to repair it.
- You can also go to Navigation menu > Compute Engine > VM instances to monitor through the console.
Congratulations!
You successfully deployed, scaled, and updated your website on Compute Engine. You are now experienced with Compute Engine, Managed Instance Groups, Load Balancers, and Health Checks!
Next steps / Learn more
Google Cloud training and certification
...helps you make the most of Google Cloud technologies. Our classes include technical skills and best practices to help you get up to speed quickly and continue your learning journey. We offer fundamental to advanced level training, with on-demand, live, and virtual options to suit your busy schedule. Certifications help you validate and prove your skill and expertise in Google Cloud technologies.
Manual Last Updated August 25, 2025
Lab Last Tested August 25, 2025
Copyright 2025 Google LLC. All rights reserved. Google and the Google logo are trademarks of Google LLC. All other company and product names may be trademarks of the respective companies with which they are associated.