Skip to content

Instantly share code, notes, and snippets.

@brianonn
Last active March 13, 2025 09:58
Show Gist options
  • Save brianonn/6c222c36d8306813851e4aa0f80b3955 to your computer and use it in GitHub Desktop.
Save brianonn/6c222c36d8306813851e4aa0f80b3955 to your computer and use it in GitHub Desktop.
delete all resources in a Google Cloud Private Network (aka VPC)

Usage

$ delete-gcloud-network.sh ny-vpc1

Below is the output

#!/usr/bin/env bash

##
## This script will delete all resources in the Google Cloud network (aka VPC) named:  ny-vpc1
##
## Name: ny-vpc1
## Description: ''
## Created-At: 2022-09-19T23:53:08.620-07:00
## Subnetworks:
##   Region               | Name                
##   ==================== | ====================
##   us-west2             | ny-sub-12           
##   us-west2             | w2-proxy-subnet     
##   us-central1          | ny-vpc1-sub111      
##   us-central1          | ny-sub-11           

set -vx

##
## delete all INTERNAL Forwarding Rules in ny-vpc1
##
gcloud -q compute forwarding-rules delete fe-w2-alb-ny-1 --region us-west2

##
## delete all Forwarding Rules in ny-vpc1
##
gcloud -q compute forwarding-rules delete alb1fe1 --global
gcloud -q compute forwarding-rules delete alb2fe1 --global
gcloud -q compute forwarding-rules delete fe-3-classic-8080 --global
gcloud -q compute forwarding-rules delete fe-w2-alb-ny-1 --region us-west2

##
## delete all Target HTTP Proxies in ny-vpc1
##
gcloud -q compute target-http-proxies delete alb1t1ny-target-proxy --global
gcloud -q compute target-http-proxies delete alb2t2ny-target-proxy --global
gcloud -q compute target-http-proxies delete classic-alb-ny-target-proxy --global
gcloud -q compute target-http-proxies delete internal-alb-ny-target-proxy --region us-west2

##
## delete all Url Maps in ny-vpc1
##
gcloud -q compute url-maps delete alb1t1ny --global
gcloud -q compute url-maps delete alb2t2ny --global
gcloud -q compute url-maps delete classic-alb-ny --global
gcloud -q compute url-maps delete internal-alb-ny --region us-west2

##
## delete all Backend Services in ny-vpc1
##
gcloud -q compute backend-services delete alb1be1 --global
gcloud -q compute backend-services delete alb2be1 --global
gcloud -q compute backend-services delete be-w2-alb --region us-west2
gcloud -q compute backend-services delete c-be-120-1 --global

##
## delete all Managed and Unmanaged Instance Groups in ny-vpc1
##
gcloud -q compute instance-groups unmanaged delete umig2t2any --zone us-central1-c
gcloud -q compute instance-groups unmanaged delete umig4t2any --zone us-west2-a
gcloud -q compute instance-groups managed delete mig-c1-vpc1 --zone us-central1-a
gcloud -q compute instance-groups managed delete mig-w2-ny --zone us-west2-a
gcloud -q compute instance-groups managed delete mig001t2any --zone us-central1-a
gcloud -q compute instance-groups managed delete mig003t2any --zone us-central1-a
gcloud -q compute instance-groups managed delete mig101t1any --zone us-central1-a
gcloud -q compute instance-groups managed delete ny-mig-instance-group-for-http-lb --zone us-central1-a

##
## delete all Instance Templates in ny-vpc1
##
gcloud -q compute instance-templates delete instance-template-ny-http-for-lb-1
gcloud -q compute instance-templates delete mig001t2any
gcloud -q compute instance-templates delete mig003t2any
gcloud -q compute instance-templates delete mig101t1any
gcloud -q compute instance-templates delete top-w2-1

##
## delete all Instances in ny-vpc1
##
gcloud -q compute instances delete --delete-disks=all mig-c1-vpc1-2njh --zone=us-central1-a
gcloud -q compute instances delete --delete-disks=all mig001t2any-ds7w --zone=us-central1-a
gcloud -q compute instances delete --delete-disks=all mig001t2any-xj58 --zone=us-central1-a
gcloud -q compute instances delete --delete-disks=all mig003t2any-jf9d --zone=us-central1-a
gcloud -q compute instances delete --delete-disks=all mig101t1any-tl2m --zone=us-central1-a
gcloud -q compute instances delete --delete-disks=all ne-vm-201-ny --zone=us-central1-a
gcloud -q compute instances delete --delete-disks=all ny-mig-instance-group-for-http-lb-2tgv --zone=us-central1-a
gcloud -q compute instances delete --delete-disks=all ny-mig-instance-group-for-http-lb-7zwp --zone=us-central1-a
gcloud -q compute instances delete --delete-disks=all vm11t2any --zone=us-central1-c
gcloud compute instances update vm12t2any --zone=us-central1-c --no-deletion-protection  # disable delete-protection on 'vm12t2any'
gcloud -q compute instances delete --delete-disks=all vm12t2any --zone=us-central1-c
gcloud -q compute instances delete --delete-disks=all vm13t2any --zone=us-central1-c
gcloud -q compute instances delete --delete-disks=all vm14t2any --zone=us-central1-c
gcloud -q compute instances delete --delete-disks=all vm3t2any --zone=us-central1-c
gcloud -q compute instances delete --delete-disks=all vm4t2any --zone=us-central1-c
gcloud -q compute instances delete --delete-disks=all mig-w2-ny-1fsj --zone=us-west2-a
gcloud -q compute instances delete --delete-disks=all vm15t2any --zone=us-west2-a
gcloud -q compute instances delete --delete-disks=all vm6t2any --zone=us-west2-a
gcloud -q compute instances delete --delete-disks=all vm7t2any --zone=us-west2-a

##
## delete all Firewall Rules in ny-vpc1
##
gcloud -q compute firewall-rules delete ny-vpc1-allow-http
gcloud -q compute firewall-rules delete ny-vpc1-allow-https
gcloud -q compute firewall-rules delete permit-ssh-in-ny-vpc1
gcloud -q compute firewall-rules delete r1-t1a-tag8022
gcloud -q compute firewall-rules delete r101-t2a-tag22sip
gcloud -q compute firewall-rules delete r11-t1a-tag22all
gcloud -q compute firewall-rules delete r11-t2a-tag80
gcloud -q compute firewall-rules delete r12-t2a-tag9090
gcloud -q compute firewall-rules delete top2-fw-rule-nyvpc1

##
## delete all Route Tables in ny-vpc1
##
gcloud -q compute routes delete default-route-0a9f0183dda0be21
gcloud -q compute routes delete default-route-0eff2bc9d8ac30d8
gcloud -q compute routes delete default-route-4315d04b5ccf5b04
gcloud -q compute routes delete default-route-e321dc5ce3caf0d0
gcloud -q compute routes delete default-route-f96b82b51573d4c9
gcloud -q compute routes delete default-route-fbefaff344d37a1b

##
## delete all Cloud Routers in ny-vpc1
##
gcloud -q compute routers delete ny-nat-rt --region us-central1

##
## delete all Subnets in ny-vpc1
##
gcloud -q compute networks subnets delete ny-sub-11 --region us-central1
gcloud -q compute networks subnets delete ny-vpc1-sub111 --region us-central1
gcloud -q compute networks subnets delete ny-sub-12 --region us-west2
gcloud -q compute networks subnets delete w2-proxy-subnet --region us-west2

##
## delete the network (VPC)
##
gcloud -q compute networks delete ny-vpc1
#!/bin/bash
## Delete an entire Google Cloud network resource (aka VPC) and all child resources associated with it
##
## This script is safe. it will not actually delete anything.
## It only outputs a series of gcloud delete commands that you can inspect and run yourself.
set -euo pipefail
function required() {
local missing=""
while [[ $# -gt 0 ]] ; do
local cmd="$1"
if ! \command -v $cmd &>/dev/null ; then
missing="$cmd $missing"
fi
shift
done
if [[ -n "$missing" ]]; then
echo "ERROR: This script requires the following missing commands: $missing"
echo " Please install these missing commands and try again."
exit 2
fi
}
function comment() {
echo
echo "##"
echo "## $*"
echo "##"
}
# require jq and gcloud
required jq gcloud
if [[ "$#" != "1" ]]; then
echo "Usage: $(basename $0) network"
exit 1
fi
NETWORK="$1"
# first get the network information for display purposes
network_json=$(gcloud compute networks describe ${NETWORK} --format json --verbosity error)
if [[ $? -ne 0 ]]; then
exit $?
fi
echo "#!/usr/bin/env bash"
comment "This script will delete all resources in the Google Cloud network (aka VPC) named: $NETWORK"
echo "$network_json" | jq -cr '"\(.name) \(.description|@sh) \(.creationTimestamp)"' \
| while read name description creationTimestamp; do \
echo "## Name: $name" ; \
echo "## Description: $description" ; \
echo "## Created-At: $creationTimestamp" ; \
done
echo "## Subnetworks:"
printf "## %-20s | %-20s\n" Region Name
printf "## %-20s | %-20s\n" ==================== ====================
echo "$network_json" | jq -cr '.subnetworks[]|"\(split("/")[-3]) \(split("/")[-1])"' \
| while read region name ; do \
printf "## %-20s | %-20s\n" $region $name ; \
done
echo
echo "set -vx"
# get and script a deletion of all internal forwarding rules on the chosen network (LB)
j=$(gcloud compute forwarding-rules list --filter="network ~ /networks/${NETWORK} AND loadBalancingScheme ~ INTERNAL" --format json --verbosity error)
count=$(echo $j | jq length)
if [[ $count -gt 0 ]] ; then
comment "delete all INTERNAL Forwarding Rules in $NETWORK"
echo "$j" \
| jq -r '.[]|{name,"region":.region|split("/")[-1]}|"\(.name) \(.region)"' \
| while read name region; do \
[ "$region" == "global" ] && region="" || region="--region $region" ;\
echo gcloud -q compute forwarding-rules delete $name $region ; \
done
fi
j='' count=''
# We can't delete instance groups that refer to backend services (and BE services might refer to an urlmap )
# so we collect them all and delete url-maps, then backend services then instance groups in that order
j=$(gcloud compute instance-groups list --filter="network ~ /networks/${NETWORK}" --format json --verbosity error)
count=$(echo "$j" | jq length)
if [[ $count -gt 0 ]] ; then
echo "$j" \
| jq -cr '.[]|{name,isManaged,"zone":.zone|split("/")[-1]}|"\(.name) \(.isManaged) \(.zone)"' \
|{ \
declare -a FWRULE HTTPTARGETS URLMAPS BESERVICES INSTGROUPS ;\
FWDRULE=() ; \
HTTPTARGETS=() ; \
URLMAPS=() ; \
BESERVICES=() ; \
INSTGROUPS=() ; \
while read name isManaged zone; do \
# find any backend services that refer to this instance group
be=$(gcloud compute backend-services list --filter='backends.group ~ instanceGroups/'${name}'' --format json --verbosity error) ;\
count=$(echo "$be" | jq length) ; \
if [[ $count -gt 0 ]]; then \
set 1 $(echo $be | jq -r '.[]|"\(.name) \(.selfLink|split("/")[-3])"') ; \
be_name="$2" ; \
be_region="$3" ; \
# find any url-maps that reference this backend-service
um=$(gcloud compute url-maps list --filter="defaultService ~ /$be_region/backendServices/$be_name OR pathMatchers.defaultService ~ /$be_region/backendServices/$be_name OR pathMatchers.pathRules.service ~ /$be_region/backendServices/$be_name" --format json --verbosity error) ;\
count=$(echo "$um" | jq length) ; \
if [[ $count -gt 0 ]]; then \
set 1 $(echo $um | jq -r '.[]|"\(.name) \(.selfLink|split("/")[-3])"') ; \
um_name="$2" ;\
um_region="$3" ;\
# find any target http proxies that reference this backend-service
ht=$(gcloud compute target-http-proxies list --filter="urlMap ~ $um_region/urlMaps/$um_name" --format json ) ;\
count=$(echo "$ht" | jq length) ; \
if [[ $count -gt 0 ]]; then \
set 1 $(echo $ht | jq -r '.[]|"\(.name) \(.selfLink|split("/")[-3])"') ; \
ht_name="$2" ;\
ht_region="$3" ;\
# find any forwarding rules that reference this httpTargetProxy
fwd=$(gcloud compute forwarding-rules list --filter="target ~ $ht_region/targetHttpProxies/$ht_name" --format json ) ;\
count=$(echo "$fwd" | jq length) ; \
if [[ $count -gt 0 ]]; then \
set 1 $(echo $fwd | jq -r '.[]|"\(.name) \(.selfLink|split("/")[-3])"') ; \
fwd_name="$2" ;\
fwd_region="$3" ;\
[ "$fwd_region" == "global" ] && fwd_region="--global" || fwd_region="--region $fwd_region" ; \
FWDRULE+=( "gcloud -q compute forwarding-rules delete $fwd_name $fwd_region" ) ; \
fi ; \
[ "$ht_region" == "global" ] && ht_region="--global" || ht_region="--region $ht_region" ; \
HTTPTARGETS+=( "gcloud -q compute target-http-proxies delete $ht_name $ht_region" ) ; \
fi ; \
[ "$um_region" == "global" ] && um_region="--global" || um_region="--region $um_region" ; \
URLMAPS+=( "gcloud -q compute url-maps delete $um_name $um_region" ) ; \
fi ;\
[ "$be_region" == "global" ] && be_region="--global" || be_region="--region $be_region" ; \
BESERVICES+=( "gcloud -q compute backend-services delete $be_name $be_region" ) ; \
fi ; \
[ "$isManaged" == "Yes" ] && managed="managed" || managed="unmanaged" ; \
INSTGROUP+=( "gcloud -q compute instance-groups $managed delete $name --zone $zone" ) ; \
done ; \
comment "delete all Forwarding Rules in $NETWORK"
for line in "${FWDRULE[@]}"; do echo $line; done | sort -u ; \
comment "delete all Target HTTP Proxies in $NETWORK"
for line in "${HTTPTARGETS[@]}"; do echo $line; done | sort -u ; \
comment "delete all Url Maps in $NETWORK"
for line in "${URLMAPS[@]}"; do echo $line; done | sort -u ; \
comment "delete all Backend Services in $NETWORK"
for line in "${BESERVICES[@]}"; do echo $line; done | sort -u ; \
comment "delete all Managed and Unmanaged Instance Groups in $NETWORK"
for line in "${INSTGROUP[@]}"; do echo $line; done | sort -u ; \
}
fi
j='' count=''
# get and script a deletion of all instance templates on the chosen network
j=$(gcloud compute instance-templates list --filter="properties.networkInterfaces.network ~ /networks/${NETWORK}" --format json --verbosity error)
count=$(echo "$j" | jq length)
if [[ $count -gt 0 ]] ; then
comment "delete all Instance Templates in $NETWORK"
echo "$j" \
| jq -r '.[]|{name,"zone":.selfLink|split("/")[-3]}|"\(.name) \(.zone)"' \
| while read name zone; do \
[ "$zone" == "global" ] && zone="" || zone="--zone $zone" ;\
echo gcloud -q compute instance-templates delete $name $zone ; \
done
fi
j='' count=''
# get and script a deletion of all instance templates on the chosen network
j=$(gcloud compute instances list --filter="networkInterfaces.network ~ /networks/${NETWORK}" --format json --verbosity error)
count=$(echo "$j" | jq length)
if [[ $count -gt 0 ]] ; then
comment "delete all Instances in $NETWORK"
echo "$j" \
| jq -r '.[]|{name,deletionProtection,"zone":.zone|split("/")[-1]}|"\(.name) \(.deletionProtection) \(.zone)"' \
| while read name deletionProtection zone; do \
if [[ -n "$deletionProtection" ]] && [[ "$deletionProtection" == "true" ]]; then \
echo "gcloud compute instances update $name --zone=$zone --no-deletion-protection # disable delete-protection on '$name'" ; \
fi ;\
echo gcloud -q compute instances delete --delete-disks=all $name --zone=$zone ; \
done
fi
j='' count=''
# get and script a deletion of all firewall rules on the chosen network
j=$(gcloud compute firewall-rules list --filter="network ~ /networks/${NETWORK}" --format json --verbosity error)
count=$(echo "$j" | jq length)
if [[ $count -gt 0 ]] ; then
comment "delete all Firewall Rules in $NETWORK"
echo "$j" \
| jq -r '.[]|{name,"zone":.selfLink|split("/")[-3]}|"\(.name) \(.zone)"' \
| while read name zone; do \
[ "$zone" == "global" ] && zone="" || zone="--zone $zone" ;\
echo gcloud -q compute firewall-rules delete $name $zone ; \
done
fi
j='' count=''
# get and script a deletion of all cloud VPN gateways on the chosen network
j=$(gcloud compute vpn-gateways list --filter="network ~ /networks/${NETWORK}" --format json --verbosity error)
count=$(echo $j | jq length)
if [[ $count -gt 0 ]] ; then
comment "delete all Cloud VPN Gateways in $NETWORK"
echo "$j" \
| jq -r '.[]|"\(.name) \(.region|split("/")[-1]"' \
| while read name region; do \
[ "$region" == "global" ] && region="" || region="--region $region" ;\
echo gcloud -q compute vpn-gateways delete $name $region ; \
done
fi
j='' count=''
# get and script a deletion of all route tables on the chosen network
j=$(gcloud compute routes list --filter="network ~ /networks/${NETWORK}" --format json --verbosity error)
count=$(echo $j | jq length)
if [[ $count -gt 0 ]] ; then
comment "delete all Route Tables in $NETWORK"
echo "$j" \
| jq -r '.[]|{name,"region":.selfLink|split("/")[-3]}|"\(.name) \(.region)"' \
| while read name region; do \
[ "$region" == "global" ] && region="" || region="--region $region" ;\
echo gcloud -q compute routes delete $name $region ; \
done
fi
j='' count=''
# get and script a deletion of all cloud routers on the chosen network
j=$(gcloud compute routers list --filter="network ~ /networks/${NETWORK}" --format json --verbosity error)
count=$(echo $j | jq length)
if [[ $count -gt 0 ]] ; then
comment "delete all Cloud Routers in $NETWORK"
echo "$j" \
| jq -r '.[]|"\(.name) \(.region|split("/")[-1])"' \
| while read name region; do \
[ "$region" == "global" ] && region="" || region="--region $region" ;\
echo gcloud -q compute routers delete $name $region ; \
done
fi
j='' count=''
# get and script a deletion of all subnets on the chosen network
j=$(gcloud compute networks subnets list --filter="network ~ /networks/${NETWORK}" --format json --verbosity error)
count=$(echo $j | jq length)
if [[ $count -gt 0 ]] ; then
comment "delete all Subnets in $NETWORK"
echo "$j" \
| jq -r '.[]|"\(.name) \(.region|split("/")[-1])"' \
| while read name region; do \
[ "$region" == "global" ] && region="" || region="--region $region" ;\
echo gcloud -q compute networks subnets delete $name $region ; \
done
fi
j='' count=''
# finally delete the network itself
comment "delete the network (VPC)"
echo gcloud -q compute networks delete $NETWORK
## Other Resources:
## Check for any other resources that might be referencing the
## network (e.g., load balancers, Cloud SQL instances, etc.) and delete them accordingly.
##
## after deleting the network, there will be broken forwarding rules, urlMaps and Backend Services that point
## to non-existant instances or instance groups
##
## cleaning that mess up is for another day
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment