-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathdeploy.sh
executable file
·250 lines (222 loc) · 9.49 KB
/
deploy.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
#!/bin/bash
# deploy.sh - build, upload sourcemaps, and deploy or run locally any one or multiple projects
# using ./env-config/*.env for both build-time and runtime environment configuration. Must be
# called with --env parameter that specifies which env-config file to use. A special case is
# --env=local which will run each project's 'run.sh' script instead of deploying to Google App
# Engine (default).
#
# Usage: ./deploy.sh react flask --env=staging
#
# Can be run from any directory. In --env=local mode it will start web servers for ALL projects
# specified on command-line (backends first) and then eventually, upon exit (Ctrl+C), every one
# of them. An error anywhere in the script (e.g. failed deployment or sentry-cli call) should
# make the entire script exit immediately.
#
# All variables in *.env are passed into each projects runtime environment.
# Some variables, however, are special and are additionally used during the build:
#
# SENTRY_ORG is passed to ./bin/sentry-release.sh and used to specify which Sentry
# org to create release in.
#
# <PROJECT>_SENTRY_PROJECT is passed to ./bin/sentry-release.sh and used to specify
# which Sentry project to create release in.
#
# <PROJECT>_RELEASE_PACKAGE_NAME is used to create a new Sentry release (prepended to calendar
# version e.g. '[email protected]' )
#
# <PROJECT>_SOURCEMAPS_URL_PREFIX
#
# <PROJECT>_SOURCEMAPS_DIR
#
# <PROJECT>_APP_ENGINE_SERVICE is used to parametrize app.yaml.template for that project and
# points to the Google App Engine (GAE) "service" to which the project should be deployed.
# These values should be configured once in production.env and staging.env and never changed
# again. Obviously, for local.env they are meaningless and not required.
# e.g.: REACT_APP_ENGINE_SERVICE
#
# REACT_APP_<PROJECT>_BACKEND besides being passed into React application runtime these are
# substituted with the values of respective REACT_APP_<PROJECT>_BACKEND_LOCAL when --env=local
# AND <PROJECT> is included in the list of projects to build from command-line arguments.
#
# MIGRATION NOTE:
#
# The old .env and app.yaml must now be deleted from ./<project>/ directory. Instead move all the
# variables to ./env-config/<env>.env and create ./<project>/app.yaml.template with '<SERVICE>' placeholder
# in place of actual service name. Finally add <PROJECT>_APP_ENGINE_SERVICE=<actual service name> to all the
# ./env-config/<env>.env files.
set -e # exit immediately if any command exits with a non-zero status
# https://fvue.nl/wiki/Bash:_Error_handling
# use top-level directory (repository root), to ensure this works regardless of current directory
top=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )
export PATH="$PATH:$top/bin"
# Parse CLI arguments
projects=""
env=""
for arg in "$@"; do
if [[ $arg = --env=* ]]; then
env=$(echo $arg | cut -d '=' -f 2)
echo "env = $env"
else
projects+="$arg "
fi
done
USAGE="[ERROR] Invalid arguments. Usage e.g.: ./deploy.sh --env=staging react flask"
# Validate CLI arguments
if [[ "$env" == "" || "$projects" == "" ]]; then
echo "$USAGE";
exit 1
fi
if [ "$env" == "production" ]; then
if [ -t 0 ] ; then # shell is interactive
verify_latest_code.sh
fi
fi
# Re-order so backends are launched first
be_projects=""
fe_projects=""
for proj in $projects; do
if [[ $proj =~ ^(flask|express|ruby|spring-boot|aspnetcore|laravel|ruby-on-rails|crons-python)$ ]]; then
be_projects+="$proj "
else
fe_projects+="$proj "
fi
done
projects="$be_projects $fe_projects"
function cleanup {
for pid in $run_sh_pids; do
# each run.sh has it's own cleanup function
if ps -p $pid > /dev/null; then
kill $pid 2>/dev/null
fi
done
rm -f $top/*/.app.yaml
rm -f $top/spring-boot/src/main/appengine/app.yaml
rm -f $top/spring-boot/src/main/resources/application.properties
rm -f $top/crons-python/crontab
rm -f $top/react/config-overrides.js
if [ "$generated_envs" != "" ]; then
rm -f $generated_envs # bash only (passed as separate args)
fi
}
trap cleanup EXIT
run_sh_pids=""
generated_envs=""
for proj in $projects; do # bash only
echo "|||"
echo "||| $0: $proj"
echo "|||"
validate_project.sh $top/$proj
cd $top/$proj
# React bakes in (exported) env variables from calling shell as well as contents of .env
# at build time into the static build output. As a result it doesn't need .env at runtime.
# See: https://github.com/facebook/create-react-app/blob/main/packages/react-scripts/config/env.js
# and https://create-react-app.dev/docs/adding-custom-environment-variables/
#
# Express and Flask on the other hand need .env deployed and present at runtime.
#
# We generate a temporary .env dynamically from env-config/*.env then remove upon exit
generated_envs+="$(../env.sh $env) "
# We do this because 1) we need RELEASE that's generated in env.sh 2) we need *_APP_*_BACKEND
# 3) some projects may require env variables instead of .env (not the case for react, flask & express)
# TODO: double check above comment is still correct, we do this 3 times (once here and twice in env.sh)
# TODO: support spring-boot which seems to use .properties files
export $(grep -v '^#' .env | sed 's/ #.*//' | xargs)
if [[ "$env" == "local" && "$fe_projects" = *"$proj "* ]]; then
# Point to local backend http://host:port instead of cloud endpoints for all _built_ BE projects
# If no backend projects specified in CLI args, keep using cloud (production or staging) BE endpoints.
for be_proj in $be_projects; do
if [ "$proj" == "next" ]; then
# Next env variables need to start with NEXT_PUBLIC_*
backend_var=$(var_name.sh NEXT_PUBLIC_%s_BACKEND $be_proj)
else
backend_var=$(var_name.sh %s_APP_%s_BACKEND $proj $be_proj)
fi
. get_proj_var.sh "%s_LOCAL_PORT" $be_proj # sets $local_port
echo "" >> .env # in case no newline
backend_local="http://localhost:$local_port"
echo "$backend_var=$backend_local" >> .env # append instead of search-replace should be OK
export "$backend_var=$backend_local"
done
fi
CLOUD_SQL_AUTH_PROXY=172.17.0.1
if [ "$env" != "local" ]; then
if [ "$proj" == "aspnetcore" ]; then
# https://cloud.google.com/sql/docs/postgres/connect-app-engine-flexible
export DB_HOST="$CLOUD_SQL_AUTH_PROXY"
elif [ "$proj" == "laravel" ]; then
# https://cloud.google.com/sql/docs/postgres/connect-app-engine-standard#php
export DB_HOST="pgsql:dbname=$DB_DATABASE;host=/cloudsql/$DB_CLOUD_SQL_CONNECTION_NAME/"
fi
fi
unset CI # prevents build failing in GitHub Actions
./build.sh
if [[ "$fe_projects" = *"$proj "* ]]; then # project is frontend
if [[ "$proj" == "react" || "$proj" == "next" ]]; then
upload_sourcemaps="false" # using webpack plugin
else
upload_sourcemaps="true"
fi
sentry-release.sh $env $RELEASE $upload_sourcemaps
# NOTE: Sentry may create releases from events even without this step
fi
# If gcloud is installed, use it to get the sentry auth token from Google Cloud Secret Manager.
# Sentry CLI will use this token for authentication. Otherwise, one can use `sentry-cli login`,
# but that will not work when deploying to staging or production.
if command -v gcloud &> /dev/null ; then
export SENTRY_AUTH_TOKEN=$(gcloud secrets versions access latest --secret="SENTRY_AUTH_TOKEN")
fi
# *** DEPLOY OR RUN ***
if [ "$env" == "local" ]; then
if [[ "$be_projects" = *"$proj "* ]]; then
. get_proj_var.sh "%s_LOCAL_PORT" $proj # sets $local_port
export LOCAL_PORT="$local_port"
fi
./run.sh &
pid="$!"
run_sh_pids+="$pid " # for later cleanup
if [[ "$projects" != *"$proj " ]]; then # not last one
sleep 1
echo "$0: Waiting a few seconds before building next project to make sure this server process doesn't crash..."
sleep 4
if ! ps -p $pid > /dev/null
then
echo "$0 [ERROR]: $proj/run.sh exited early, must be a crash."
exit 1
fi
fi
elif [ -f deploy_project.sh ]; then
if [[ "$proj" =~ ^crons- ]]; then
. get_proj_var.sh "%s_DEPLOY_DIR" $proj
escaped_deploy_dir=$(echo "$deploy_dir" | sed 's_/_\\/_g')
sed -e 's/<CRONSPYTHON_DEPLOY_DIR>/'$escaped_deploy_dir'/g' crontab.template > crontab
fi
./deploy_project.sh
else
# Replace <SERVICE> in app.yaml.template with <PROJECT>_APP_ENGINE_SERVICE
. get_proj_var.sh "%s_APP_ENGINE_SERVICE" $proj
if [ "$proj" == "spring-boot" ]; then
ypath="./src/main/appengine/"
sed -e 's/<SERVICE>/'$app_engine_service'/g' $ypath/app.yaml.template > $ypath/app.yaml
# This sets SPRINGBOOT_ENV
# TODO: Un-hardcode. Q: what non-production values does it take?
sed_inplace.sh 's/<ENV>/"production"/g' $ypath/app.yaml
sed_inplace.sh 's/<CLOUD SQL CONNECTION NAME>/"'"$DB_CLOUD_SQL_CONNECTION_NAME"'"/g' $ypath/app.yaml
mvn clean package appengine:deploy
elif [ "$proj" == "aspnetcore" ]; then
# TODO: envsubst is super easy - this should be the default for all projects
envsubst < app.yaml.template > .app.yaml
envsubst < Dockerfile.template > Dockerfile
gcloud app deploy --version v1 --quiet .app.yaml
rm Dockerfile
else
# all other projects
sed -e 's/<SERVICE>/'$app_engine_service'/g' app.yaml.template > .app.yaml
gcloud app deploy --version v1 --quiet .app.yaml
fi
fi
done
if [ "$env" == "local" ]; then
sleep 1
echo "Server process(es) are running. Press Ctrl+C to terminate..."
while true; do sleep 86400; done
fi