forked from GoogleCloudPlatform/software-delivery-blueprint
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsoftware-delivery-infra.sh
executable file
·562 lines (491 loc) · 29.2 KB
/
software-delivery-infra.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
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
#!/usr/bin/env bash
shopt -s expand_aliases
# Copyright 2022 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
#
# http://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.
# Verify that the scripts are being run from Linux and not Mac
if [[ $OSTYPE != "linux-gnu" ]]; then
echo "ERROR: This script and consecutive set up scripts have only been tested on Linux. Currently, only Linux (debian) is supported. Please run in Cloud Shell or in a VM running Linux".
exit;
fi
export SCRIPT_DIR=$(dirname $(readlink -f $0 2>/dev/null) 2>/dev/null || echo "${PWD}/$(dirname $0)")
START_DIR=${PWD}
BASE_DIR="${SCRIPT_DIR}/../"
LOG_DIR="${SCRIPT_DIR}/../../logs/infra"
mkdir -p ${LOG_DIR}
TEMP_DIR="${BASE_DIR}/../temp"
mkdir -p ${TEMP_DIR}
if [ ! -f ${LOG_DIR}/vars.sh ]; then
cp ${SCRIPT_DIR}/vars.sh ${LOG_DIR}/vars.sh
else
source ${LOG_DIR}/vars.sh
fi
export LOG_FILE=${LOG_DIR}/platform-bootstrap-$(date +%s).log
touch ${LOG_FILE}
exec 2>&1
exec &> >(tee -i ${LOG_FILE})
#functions.sh helps make the script interactive
source ${SCRIPT_DIR}/functions.sh
# Ensure Org ID is defined otherwise collect
while [ -z ${ORG_NAME} ]
do
read -p "$(echo -e "Please provide your Organization Name (your active account must be Org Admin): ")" ORG_NAME
done
# Validate ORG_NAME exists
ORG_ID=$(gcloud organizations list \
--filter="display_name=${ORG_NAME}" \
--format="value(ID)")
[ ${ORG_ID} ] || { echo "Organization with that name does not exist or you do not have correct permissions in this org."; exit; }
# Ensure Billing account is defined otherwise collect
while [ -z ${BILLING_ACCOUNT_ID} ]
do
read -p "$(echo -e "Please provide your Billing Account ID (your active account must be Billing Account Admin): ")" BILLING_ACCOUNT_ID
done
# Check if FOLDER_NAME is needed. If not, enter just press enter
read -p "$(echo -e "Please provide Folder Name (your active account must be Org Admin): ")" FOLDER_NAME
# Ensure infra setup project name is defined
while [ -z ${INFRA_SETUP_PROJECT} ]
do
read -p "$(echo -e "Please provide the name for multi-tenant admin project: ")" INFRA_SETUP_PROJECT
done
# Ensure infra setup repo name is defined
while [ -z ${INFRA_SETUP_REPO} ]
do
read -p "$(echo -e "Please provide the name for multi-tenant platform repo: ")" INFRA_SETUP_REPO
done
# Ensure github user is defined
while [ -z ${GITHUB_USER} ]
do
read -p "$(echo -e "Please provide your github user: ")" GITHUB_USER
done
# Ensure github personal access token is defined
while [ -z ${TOKEN} ]
do
read -p "$(echo -e "Please provide your github personal access token: ")" TOKEN
done
# Ensure github org is defined
while [ -z ${GITHUB_ORG} ]
do
read -p "$(echo -e "Please provide your github org: ")" GITHUB_ORG
done
# Ensure REGION is defined
while [ -z ${REGION} ]
do
read -p "$(echo -e "Please provide the region where resources will be created: ")" REGION
done
# Let the user chose the kind of trigger they want to create in Application factory
while [ -z ${TRIGGER_TYPE} ]
do
read -p "$(echo -e "Enter \"webhook\" for creating webhook triggers or \"github\" for github trigger: ")" TRIGGER_TYPE
done
# Ensure IAM group name is defined
while [ -z ${IAM_GROUP} ]
do
read -p "$(echo -e "Please provide a name for DevOps IAM group: ")" IAM_GROUP
done
TEMPLATE_INFRA_REPO="platform-template"
TEMPLATE_ACM_REPO="acm-template"
ACM_REPO="acm-${INFRA_SETUP_REPO}"
TF_MODULES="terraform-modules"
TIMESTAMP=$(date "+%Y%m%d%H%M%S")
SA_FOR_API_KEY="api-key-sa-${TIMESTAMP}"
IAM_GROUP_ALREADY_EXISTS=0
PROJECT_ID_SUFFIX=$(cat /dev/urandom | tr -dc 'a-z0-9' | fold -w 6 | head -n 1)
if [[ -z ${INFRA_SETUP_PROJECT_ID} ]]; then
INFRA_SETUP_PROJECT_ID=${INFRA_SETUP_PROJECT}-${PROJECT_ID_SUFFIX}
fi
#Storing variables in the state file so the script start from where it left off in even of a failure
grep -q "export INFRA_SETUP_PROJECT_ID.*" ${LOG_DIR}/vars.sh || echo -e "export INFRA_SETUP_PROJECT_ID=${INFRA_SETUP_PROJECT_ID}" >> ${LOG_DIR}/vars.sh
grep -q "export INFRA_SETUP_PROJECT=.*" ${LOG_DIR}/vars.sh || echo -e "export INFRA_SETUP_PROJECT=${INFRA_SETUP_PROJECT}" >> ${LOG_DIR}/vars.sh
grep -q "export BILLING_ACCOUNT_ID=.*" ${LOG_DIR}/vars.sh|| echo -e "export BILLING_ACCOUNT_ID=${BILLING_ACCOUNT_ID}" >> ${LOG_DIR}/vars.sh
grep -q "export ORG_NAME=.*" ${LOG_DIR}/vars.sh|| echo -e "export ORG_NAME=${ORG_NAME}" >> ${LOG_DIR}/vars.sh
grep -q "export ORG_ID=.*" ${LOG_DIR}/vars.sh || echo -e "export ORG_ID=${ORG_ID}" >> ${LOG_DIR}/vars.sh
grep -q "export FOLDER_NAME=.*" ${LOG_DIR}/vars.sh|| echo -e "export FOLDER_NAME=${FOLDER_NAME}" >> ${LOG_DIR}/vars.sh
grep -q "export FOLDER_ID=.*" ${LOG_DIR}/vars.sh || echo -e "export FOLDER_ID=${FOLDER_ID}" >> ${LOG_DIR}/vars.sh
grep -q "export INFRA_SETUP_REPO=.*" ${LOG_DIR}/vars.sh || echo -e "export INFRA_SETUP_REPO=${INFRA_SETUP_REPO}" >> ${LOG_DIR}/vars.sh
grep -q "export GITHUB_USER=.*" ${LOG_DIR}/vars.sh || echo -e "export GITHUB_USER=${GITHUB_USER}" >> ${LOG_DIR}/vars.sh
grep -q "export TOKEN=.*" ${LOG_DIR}/vars.sh|| echo -e "export TOKEN=${TOKEN}" >> ${LOG_DIR}/vars.sh
grep -q "export GITHUB_ORG=.*" ${LOG_DIR}/vars.sh || echo -e "export GITHUB_ORG=${GITHUB_ORG}" >> ${LOG_DIR}/vars.sh
grep -q "export REGION=.*" ${LOG_DIR}/vars.sh || echo -e "export REGION=${REGION}" >> ${LOG_DIR}/vars.sh
if [[ "${TRIGGER_TYPE,,}" == "webhook" ]] || [[ "${TRIGGER_TYPE,,}" == "github" ]]; then
grep -q "export TRIGGER_TYPE=.*" ${LOG_DIR}/vars.sh || echo -e "export TRIGGER_TYPE=${TRIGGER_TYPE}" >> ${LOG_DIR}/vars.sh
else
title_no_wait "The trigger type ${TRIGGER_TYPE} is invalid. Specify webhook or github"
exit 1
fi
#FUNCTIONS DEFINITIONS BELOW
generate_api_key () {
title_no_wait "Creaing a Service Account for creating an API key ..."
print_and_execute "gcloud iam service-accounts create ${SA_FOR_API_KEY} --display-name \"API Key {SA_FOR_API_KEY}\""
title_no_wait "Granting access to cloudbuild SA for accessing the API key ..."
print_and_execute "gcloud projects add-iam-policy-binding ${INFRA_SETUP_PROJECT_ID} \
--member serviceAccount:${SA_FOR_API_KEY}@${INFRA_SETUP_PROJECT_ID}.iam.gserviceaccount.com \
--role roles/serviceusage.apiKeysAdmin"
title_no_wait "Creating credentials for the SA ..."
print_and_execute "gcloud iam service-accounts keys create ~/credentials.json \
--iam-account ${SA_FOR_API_KEY}@${INFRA_SETUP_PROJECT_ID}.iam.gserviceaccount.com"
if [[ `which oauth2l | wc -l` -eq 0 ]]; then
title_no_wait "oauth2l not installed"
title_no_wait "Download and install oauth2l ..."
print_and_execute "git clone https://${GITHUB_USER}:${TOKEN}@github.com/google/oauth2l ${START_DIR}/oauth2l-${TIMESTAMP}"
print_and_execute "cd ${START_DIR}/oauth2l-${TIMESTAMP}"
print_and_execute "make dev"
print_and_execute "oauth2l fetch --credentials ~/credentials.json --scope cloud-platform"
print_and_execute "alias gcurl='curl -S -H \"$(oauth2l header --json ~/credentials.json cloud-platform userinfo.email)\" -H \"Content-Type: application/json\"'"
else
title_no_wait "oauth2l is already installed ..."
print_and_execute "alias gcurl='curl -S -H \"$(oauth2l header --json ~/credentials.json cloud-platform userinfo.email)\" -H \"Content-Type: application/json\"'"
fi
title_no_wait "checking if we are all set to create API key ..."
print_and_execute "type gcurl"
if [ $? -ne 0 ]; then
title_no_wait "gcurl alias not set. Problem in using oauth2l. Exiting"
exit 1
fi
title_no_wait "Creating a API key for webhook ..."
print_and_execute "operation_id=$(gcurl https://apikeys.googleapis.com/v2/projects/${INFRA_PROJECT_NUMBER}/locations/global/keys -X POST -d '{"displayName" : "webhook","restrictions": {"api_targets": [{"service": "cloudbuild.googleapis.com"}]}}' | jq .name)"
title_no_wait "Polling for the operation to complete ..."
status="false"
while [ ${status} != "true" ]
do
title_no_wait "Waiting for operation to create API to complete. Sleeping for 5"
print_and_execute "status=$(gcurl https://apikeys.googleapis.com/v2/${operation_id} | jq .done)"
if [[ -z ${status} ]]; then
title_no_wait "Unable to fetch the status of API key create operation. Possibly the command to create API key had issues. Aborting"
exit
fi
sleep 5
done
print_and_execute "API_KEY=$(gcurl https://apikeys.googleapis.com/v2/${operation_id} | jq .response.keyString)"
}
create_webhook () {
title_no_wait "Creating a secret for webhook ..."
SECRET_NAME=webhook-secret-${TIMESTAMP}
SECRET_VALUE=$(sed "s/[^a-zA-Z0-9]//g" <<< $(openssl rand -base64 15))
SECRET_PATH=projects/${INFRA_PROJECT_NUMBER}/secrets/${SECRET_NAME}/versions/1
print_and_execute "printf ${SECRET_VALUE} | gcloud secrets create ${SECRET_NAME} --data-file=-"
title_no_wait "Providing read access to Cloudbuild service account on the secret ..."
print_and_execute "gcloud secrets add-iam-policy-binding ${SECRET_NAME} \
--member=serviceAccount:service-${INFRA_PROJECT_NUMBER}@gcp-sa-cloudbuild.iam.gserviceaccount.com \
--role='roles/secretmanager.secretAccessor'"
title_no_wait "Creating a webhook trigger ..."
print_and_execute "gcloud alpha builds triggers create webhook --name=\"infra-trigger-${TIMESTAMP}\" --inline-config=\"${TEMP_DIR}/${INFRA_SETUP_REPO}/cloudbuild-webhook.yaml\" --secret=${SECRET_PATH} --substitutions='_REF=\$(body.ref),_REPO=\$(body.repository.full_name),_COMMIT_MSG=\$(body.head_commit.message)' --subscription-filter='(!_COMMIT_MSG.matches(\"IGNORE\"))'"
# Retrieve the URL
WEBHOOK_URL="https://cloudbuild.googleapis.com/v1/projects/${INFRA_SETUP_PROJECT_ID}/triggers/infra-trigger-${TIMESTAMP}:webhook?key=${API_KEY}&secret=${SECRET_VALUE}"
title_no_wait "Creating a github webhook ..."
print_and_execute "curl -H \"Authorization: token ${TOKEN}\" \
-d '{\"config\": {\"url\": \"${WEBHOOK_URL}\", \"content_type\": \"json\"}}' \
-X POST https://api.github.com/repos/$GITHUB_ORG/$INFRA_SETUP_REPO/hooks"
}
title_no_wait "STARTING"
# Create folder if the FOLDER_NAME was not entered blank
if [[ -n ${FOLDER_NAME} ]]; then
title_no_wait "Checking if folder ${FOLDER_NAME} already exists..."
print_and_execute "folder_flag=$(gcloud resource-manager folders list --organization ${ORG_ID} --format=json --filter="display_name=${FOLDER_NAME}" | grep "\"displayName\": \"$FOLDER_NAME\"" | wc -l)"
if [ ${folder_flag} -eq 0 ]; then
title_no_wait "Folder ${FOLDER_NAME} does not exist. Creating it..."
print_and_execute "gcloud resource-manager folders create --display-name=${FOLDER_NAME} --organization=${ORG_ID}"
FOLDER_ID=$(gcloud resource-manager folders list --organization=${ORG_ID} --filter="display_name=${FOLDER_NAME}" --format="value(ID)")
grep -q "export FOLDER_ID.*" ${LOG_DIR}/vars.sh || echo -e "export FOLDER_ID=${FOLDER_ID}" >> ${LOG_DIR}/vars.sh
else
title_no_wait "Folder ${FOLDER_NAME} already exists. Finding its ID..."
FOLDER_ID=$(gcloud resource-manager folders list --organization=${ORG_ID} --filter="display_name=${FOLDER_NAME}" --format="value(ID)")
grep -q "export FOLDER_ID.*" ${LOG_DIR}/vars.sh || echo -e "export FOLDER_ID=${FOLDER_ID}" >> ${LOG_DIR}/vars.sh
fi
fi
# Create infa setup project
title_no_wait "Checking if ${INFRA_SETUP_PROJECT_ID} exist already..."
project_id=$(gcloud projects list --filter="PROJECT_ID=${INFRA_SETUP_PROJECT_ID}" --format="value(PROJECT_ID)")
if [[ -z ${project_id} ]]; then
title_no_wait "${INFRA_SETUP_PROJECT_ID} does not exist.Creating platform setup project ..."
if [[ -n ${FOLDER_NAME} ]]; then
print_and_execute "gcloud projects create ${INFRA_SETUP_PROJECT_ID} \
--folder ${FOLDER_ID} \
--name ${INFRA_SETUP_PROJECT_ID} \
--set-as-default"
else
print_and_execute "gcloud projects create ${INFRA_SETUP_PROJECT_ID} \
--name ${INFRA_SETUP_PROJECT_ID} \
--set-as-default"
fi
else
title_no_wait "${INFRA_SETUP_PROJECT_ID} already exists, not creating it..."
fi
title_no_wait "STARTING"
title_no_wait "Adding git configs"
print_and_execute "git config --global user.email ${GITHUB_USER}@github.com"
print_and_execute "git config --global user.name ${GITHUB_USER}"
title_no_wait "Linking billing account to the ${INFRA_SETUP_PROJECT_ID}..."
print_and_execute "gcloud beta billing projects link ${INFRA_SETUP_PROJECT_ID} \
--billing-account ${BILLING_ACCOUNT_ID}"
# Creating terraform modules repo in your org and committing the code from template to it
title_no_wait "Checking if ${TF_MODULES} already exists..."
repo_id_exists=$(curl -s -H "Authorization: token ${TOKEN}" -H "Accept: application/json" "https://api.github.com/repos/${GITHUB_ORG}/${TF_MODULES}" | jq '.id')
if [ ${repo_id_exists} = "null" ]; then
title_no_wait "${TF_MODULES} does not exist. Creating it..."
print_and_execute "repo_id=$(curl -s -H "Authorization: token ${TOKEN}" -H "Accept: application/json" \
-d "{ \
\"name\": \"${TF_MODULES}\", \
\"private\": true \
}" \
-X POST https://api.github.com/orgs/${GITHUB_ORG}/repos | jq '.id')"
sleep 5
if [ ${repo_id} = "null" ]; then
echo "Unable to create git repo.Exiting"
exit 1
else
grep -q "export TF_MODULES=.*" ${LOG_DIR}/vars.sh || echo -e "export TF_MODULES=${TF_MODULES}" >> ${LOG_DIR}/vars.sh
fi
else
echo "The repo ${TF_MODULES} already exists, not creating it"
fi
title_no_wait "Cloning recently created terraform-modules repo..."
print_and_execute "rm -rf ${TEMP_DIR}/${TF_MODULES} && git clone https://${GITHUB_USER}:${TOKEN}@github.com/${GITHUB_ORG}/${TF_MODULES} ${TEMP_DIR}/${TF_MODULES}"
if [[ -d ${BASE_DIR}/${TEMPLATE_TF_MODULES} ]]; then
print_and_execute "cd ${TEMP_DIR}/${TF_MODULES}"
print_and_execute "git checkout main 2>/dev/null || git checkout -b main"
print_and_execute "cp -r ${BASE_DIR}/${TF_MODULES}/* ."
print_and_execute "git add . && git commit -m \"Adding repo\""
print_and_execute "git push -u origin main"
else
title_no_wait "Can not find ${BASE_DIR}/${TF_MODULES}. Exiting"
print_and_execute "exit 1"
fi
# Creating acm repo in your org and committing the code from template to it
title_no_wait "Checking if ${ACM_REPO} already exists..."
repo_id_exists=$(curl -s -H "Authorization: token ${TOKEN}" -H "Accept: application/json" "https://api.github.com/repos/${GITHUB_ORG}/${ACM_REPO}" | jq '.id')
if [ ${repo_id_exists} = "null" ]; then
title_no_wait "${ACM_REPO} does not exist. Creating it..."
print_and_execute "repo_id=$(curl -s -H "Authorization: token ${TOKEN}" -H "Accept: application/json" \
-d "{ \
\"name\": \"${ACM_REPO}\", \
\"private\": true \
}" \
-X POST https://api.github.com/orgs/${GITHUB_ORG}/repos | jq '.id')"
sleep 5
if [ ${repo_id} = "null" ]; then
echo "Unable to create git repo.Exiting"
exit 1
else
grep -q "export ACM_REPO=.*" ${LOG_DIR}/vars.sh || echo -e "export ACM_REPO=${ACM_REPO}" >> ${LOG_DIR}/vars.sh
fi
else
echo "The repo ${ACM_REPO} already exists, not creating it"
fi
title_no_wait "Cloning recently created acm repo..."
print_and_execute "rm -rf ${TEMP_DIR}/${ACM_REPO} && git clone https://${GITHUB_USER}:${TOKEN}@github.com/${GITHUB_ORG}/${ACM_REPO} ${TEMP_DIR}/${ACM_REPO}"
if [[ -d ${BASE_DIR}/${TEMPLATE_ACM_REPO} ]]; then
print_and_execute "cd ${TEMP_DIR}/${ACM_REPO}"
print_and_execute "git checkout dev 2>/dev/null || git checkout -b dev"
print_and_execute "cp -r ${BASE_DIR}/${TEMPLATE_ACM_REPO}/* ."
print_and_execute "git add . && git commit -m \"Adding repo\""
print_and_execute "git push -u origin dev"
title_no_wait "Pushing staging branch to ${ACM_REPO} ..."
print_and_execute "git checkout staging 2>/dev/null || git checkout -b staging"
print_and_execute "git push -u origin staging"
title_no_wait "Pushing prod branch to ${ACM_REPO} ..."
print_and_execute "git checkout prod 2>/dev/null || git checkout -b prod"
print_and_execute "git push -u origin prod"
else
title_no_wait "Can not find ${BASE_DIR}/${TEMPLATE_ACM_REPO}. Exiting"
print_and_execute "exit 1"
fi
#Secure staging and prod branch but disallowing direct push to them
title_no_wait "Applying branch protection..."
print_and_execute "curl -s -X PUT -u $GITHUB_USER:$TOKEN -H \"Accept: application/vnd.github.v3+json\" \
https://api.github.com/repos/$GITHUB_ORG/$ACM_REPO/branches/staging/protection \
-d \"{ \
\\\"restrictions\\\": null,\\\"required_status_checks\\\": null, \
\\\"required_pull_request_reviews\\\" : {\\\"dismissal_restrictions\\\": {}, \
\\\"dismiss_stale_reviews\\\": false,\\\"require_code_owner_reviews\\\": true,\
\\\"required_approving_review_count\\\": 1,\\\"bypass_pull_request_allowances\\\": {}}, \
\\\"enforce_admins\\\": true
}\" \
"
print_and_execute "curl -s -X PUT -u $GITHUB_USER:$TOKEN -H \"Accept: application/vnd.github.v3+json\" \
https://api.github.com/repos/$GITHUB_ORG/$ACM_REPO/branches/prod/protection \
-d \"{ \
\\\"restrictions\\\": null,\\\"required_status_checks\\\": null, \
\\\"required_pull_request_reviews\\\" : {\\\"dismissal_restrictions\\\": {}, \
\\\"dismiss_stale_reviews\\\": false,\\\"require_code_owner_reviews\\\": true,\
\\\"required_approving_review_count\\\": 1,\\\"bypass_pull_request_allowances\\\": {}}, \
\\\"enforce_admins\\\": true
}\" \
"
# Creating platform repo in your org and committing the code from template to it
title_no_wait "Checking if ${INFRA_SETUP_REPO} already exists..."
repo_id_exists=$(curl -s -H "Authorization: token ${TOKEN}" -H "Accept: application/json" "https://api.github.com/repos/${GITHUB_ORG}/${INFRA_SETUP_REPO}" | jq '.id')
if [ ${repo_id_exists} = "null" ]; then
title_no_wait "${INFRA_SETUP_REPO} does not exist. Creating it ..."
print_and_execute "repo_id=$(curl -s -H "Authorization: token ${TOKEN}" -H "Accept: application/json" \
-d "{ \
\"name\": \"${INFRA_SETUP_REPO}\", \
\"private\": true \
}" \
-X POST https://api.github.com/orgs/${GITHUB_ORG}/repos | jq '.id')"
sleep 5
if [ ${repo_id} = "null" ]; then
echo "Unable to create git repo.Exiting"
exit 1
else
grep -q "export INFRA_SETUP_REPO=.*" ${LOG_DIR}/vars.sh || echo -e "export INFRA_SETUP_REPO=${INFRA_SETUP_REPO}" >> ${LOG_DIR}/vars.sh
fi
else
echo "The repo ${INFRA_SETUP_REPO} already exists, not creating it"
fi
title_no_wait "Cloning recently created infra repo ${INFRA_SETUP_REPO}..."
print_and_execute "rm -rf ${TEMP_DIR}/${INFRA_SETUP_REPO} && git clone https://${GITHUB_USER}:${TOKEN}@github.com/${GITHUB_ORG}/${INFRA_SETUP_REPO} ${TEMP_DIR}/${INFRA_SETUP_REPO}"
if [[ -d ${BASE_DIR}/${TEMPLATE_INFRA_REPO} ]]; then
print_and_execute "cd ${TEMP_DIR}/${INFRA_SETUP_REPO}"
print_and_execute "git checkout dev 2>/dev/null || git checkout -b dev"
print_and_execute "cp -r ${BASE_DIR}/${TEMPLATE_INFRA_REPO}/* ."
print_and_execute "git add . && git commit -m \"Adding repo\""
print_and_execute "git push -u origin dev"
title_no_wait "Pushing staging branch to ${INFRA_SETUP_REPO} ..."
print_and_execute "git checkout staging 2>/dev/null || git checkout -b staging"
print_and_execute "git push -u origin staging"
title_no_wait "Pushing prod branch to ${INFRA_SETUP_REPO} ..."
print_and_execute "git checkout prod 2>/dev/null || git checkout -b prod"
print_and_execute "git push -u origin prod"
else
title_no_wait "Can not find ${BASE_DIR}/${TEMPLATE_INFRA_REPO}. Exiting"
print_and_execute "exit 1"
fi
#Secure staging and prod branch but disallowing direct push to them
title_no_wait "Applying branch protection..."
print_and_execute "curl -s -X PUT -u $GITHUB_USER:$TOKEN -H \"Accept: application/vnd.github.v3+json\" \
https://api.github.com/repos/$GITHUB_ORG/$INFRA_SETUP_REPO/branches/staging/protection \
-d \"{ \
\\\"restrictions\\\": null,\\\"required_status_checks\\\": null, \
\\\"required_pull_request_reviews\\\" : {\\\"dismissal_restrictions\\\": {}, \
\\\"dismiss_stale_reviews\\\": false,\\\"require_code_owner_reviews\\\": true,\
\\\"required_approving_review_count\\\": 1,\\\"bypass_pull_request_allowances\\\": {}}, \
\\\"enforce_admins\\\": true
}\" \
"
print_and_execute "curl -s -X PUT -u $GITHUB_USER:$TOKEN -H \"Accept: application/vnd.github.v3+json\" \
https://api.github.com/repos/$GITHUB_ORG/$INFRA_SETUP_REPO/branches/prod/protection \
-d \"{ \
\\\"restrictions\\\": null,\\\"required_status_checks\\\": null, \
\\\"required_pull_request_reviews\\\" : {\\\"dismissal_restrictions\\\": {}, \
\\\"dismiss_stale_reviews\\\": false,\\\"require_code_owner_reviews\\\": true,\
\\\"required_approving_review_count\\\": 1,\\\"bypass_pull_request_allowances\\\": {}}, \
\\\"enforce_admins\\\": true
}\" \
"
#Setting up the infrastructure setup project
title_no_wait "Setting project..."
print_and_execute "gcloud config set project ${INFRA_SETUP_PROJECT_ID}"
title_no_wait "Enabling APIs..."
print_and_execute "gcloud services enable cloudresourcemanager.googleapis.com \
cloudbilling.googleapis.com \
cloudbuild.googleapis.com \
iam.googleapis.com \
secretmanager.googleapis.com \
container.googleapis.com \
apikeys.googleapis.com \
cloudidentity.googleapis.com \
gkehub.googleapis.com \
anthosconfigmanagement.googleapis.com"
print_and_execute "sleep 10"
title_no_wait "Checking if IAM Group already exists..."
print_and_execute "group_id_exists=$(gcloud beta identity groups describe "${IAM_GROUP}@${ORG_NAME}" --format=json | jq '.name' | tr '"' ' ' | awk -F '/' '{print $2}')"
if [[ -z ${group_id_exists} ]]; then
title_no_wait "${IAM_GROUP} does not exist. Creating it..."
print_and_execute "gcloud beta identity groups create "${IAM_GROUP}@${ORG_NAME}" --organization=${ORG_ID}"
else
title_no_wait "IAM group ${IAM_GROUP} already exists, not creating it..."
IAM_GROUP_ALREADY_EXISTS=1
fi
title_no_wait "Fetching IAM Group ID that you created above..."
print_and_execute "GROUP_ID=$(gcloud beta identity groups describe "${IAM_GROUP}@${ORG_NAME}" --format=json | jq '.name' | tr '"' ' ' | awk -F '/' '{print $2}')"
grep -q "export IAM_GROUP=.*" ${LOG_DIR}/vars.sh || echo -e "export IAM_GROUP=${IAM_GROUP}" >> ${LOG_DIR}/vars.sh
if [ ${IAM_GROUP_ALREADY_EXISTS} -eq 0 ]; then
title_and_wait "ATTENTION : We need to update the IAM group to allow external members so we can add service accounts to it. As of now, there is no easy way of doing it automatically, press ENTER for instructions for doing it manually."
title_and_wait_step "Go to https://admin.google.com/u/2/ac/groups/${GROUP_ID} . Click Access Settings. Click the pencil icon and turn on the Allow members outside your organization setting. Click Save."
fi
title_no_wait "Granting required roles to the IAM group..."
print_and_execute "gcloud organizations add-iam-policy-binding ${ORG_ID} --member group:"${IAM_GROUP}@${ORG_NAME}" --role=roles/secretmanager.viewer --condition=None"
print_and_execute "gcloud organizations add-iam-policy-binding ${ORG_ID} --member group:"${IAM_GROUP}@${ORG_NAME}" --role=roles/secretmanager.secretAccessor --condition=None"
print_and_execute "gcloud organizations add-iam-policy-binding ${ORG_ID} --member group:"${IAM_GROUP}@${ORG_NAME}" --role=roles/storage.objectViewer --condition=None"
print_and_execute "gcloud organizations add-iam-policy-binding ${ORG_ID} --member group:"${IAM_GROUP}@${ORG_NAME}" --role=roles/container.developer --condition=None"
title_no_wait "Getting project number for ${INFRA_SETUP_PROJECT}"
print_and_execute "INFRA_PROJECT_NUMBER=$(gcloud projects describe ${INFRA_SETUP_PROJECT_ID} --format=json | jq '.projectNumber')"
title_no_wait "Add Cloud build service account as billing account user on the org"
print_and_execute "gcloud organizations add-iam-policy-binding ${ORG_ID} --member=serviceAccount:${INFRA_PROJECT_NUMBER}@cloudbuild.gserviceaccount.com --role=roles/billing.user --condition=None"
title_no_wait "Give cloudbuild service account projectCreator role at Org level..."
print_and_execute "gcloud organizations add-iam-policy-binding ${ORG_ID} --member=serviceAccount:${INFRA_PROJECT_NUMBER}@cloudbuild.gserviceaccount.com --role=roles/resourcemanager.projectCreator --condition=None"
title_no_wait "Give cloudbuild service account secretmanager admin role on project ${INFRA_SETUP_PROJECT_ID} ..."
print_and_execute "gcloud projects add-iam-policy-binding ${INFRA_SETUP_PROJECT_ID} --member=serviceAccount:${INFRA_PROJECT_NUMBER}@cloudbuild.gserviceaccount.com --role=roles/secretmanager.admin"
title_no_wait "Adding secrets to the multi-tenant admin project..."
print_and_execute "printf ${TOKEN} | gcloud secrets create github-token --data-file=-"
print_and_execute "printf ${GITHUB_USER} | gcloud secrets create github-user --data-file=-"
print_and_execute "printf ${GITHUB_USER}@github.com | gcloud secrets create github-email --data-file=-"
print_and_execute "printf ${GITHUB_ORG} | gcloud secrets create github-org --data-file=-"
print_and_execute "printf ${BILLING_ACCOUNT_ID} | gcloud secrets create gcp-billingac --data-file=-"
print_and_execute "printf ${ORG_ID} | gcloud secrets create gcp-org --data-file=-"
print_and_execute "printf ${GROUP_ID} | gcloud secrets create group-id --data-file=-"
print_and_execute "printf ${INFRA_SETUP_PROJECT_ID} | gcloud secrets create infra-project-id --data-file=-"
print_and_execute "printf ${ACM_REPO} | gcloud secrets create acm-repo --data-file=-"
if [[ -n ${FOLDER_ID} ]]; then
print_and_execute "printf ${FOLDER_ID} | gcloud secrets create gcp-folder --data-file=-"
else
print_and_execute "printf \" \" | gcloud secrets create gcp-folder --data-file=-"
fi
INFRA_TF_BUCKET="${INFRA_SETUP_PROJECT_ID}-multi-tenant-platform-tf-state"
title_no_wait "Creating GCS bucket for holding terraform state files..."
print_and_execute "gsutil mb -p ${INFRA_SETUP_PROJECT_ID} -l ${REGION} gs://${INFRA_TF_BUCKET}"
if [[ "${TRIGGER_TYPE,,}" == "webhook" ]]; then
generate_api_key
create_webhook
elif [[ "${TRIGGER_TYPE,,}" == "github" ]]; then
title_and_wait "ATTENTION : We need to connect Cloud Build in ${INFRA_SETUP_PROJECT_ID} with your github repo. As of now, there is no way of doing it automatically, press ENTER for instructions for doing it manually."
title_and_wait_step "Go to https://console.cloud.google.com/cloud-build/triggers/connect?project=${INFRA_PROJECT_NUMBER} \
Select \"Source\" as github and press continue. \
If it asks for authentication, enter your github credentials. \
Under \"Select Repository\" , on \"github account\" drop down click on \"+Add\" and choose ${GITHUB_ORG}. \
Click on \"repository\" drop down and select ${INFRA_SETUP_REPO}. \
Click the checkbox to agree to the terms and conditions and click connect. \
Click Done. \
"
title_no_wait "Creating Cloud Build trigger..."
print_and_execute "gcloud beta builds triggers create github --name=\"infra-trigger\" --repo-owner=\"${GITHUB_ORG}\" --repo-name=\"${INFRA_SETUP_REPO}\" --branch-pattern=\".*\" --build-config=\"cloudbuild-github.yaml\""
fi
#Perform sed operation to replace templated variables with real values
cd ${TEMP_DIR}/${INFRA_SETUP_REPO}
title_no_wait "Checkout dev branch..."
print_and_execute "git checkout dev"
title_no_wait "Replacing variables in variables.tf under dev, staging and prod folder in ${INFRA_SETUP_REPO}..."
print_and_execute "sed -i \"s/YOUR_IAM_GROUP/${IAM_GROUP}@${ORG_NAME}/\" env/*/variables.tf"
print_and_execute "sed -i \"s/YOUR_BILLING_ACCOUNT/${BILLING_ACCOUNT_ID}/\" env/*/variables.tf"
print_and_execute "sed -i \"s/YOUR_ORG_ID/${ORG_ID}/\" env/*/variables.tf"
print_and_execute "sed -i \"s/YOUR_ACM_REPO/${ACM_REPO}/\" env/*/variables.tf"
print_and_execute "find . -type f -exec sed -i \"s/YOUR_GITHUB_ORG/${GITHUB_ORG}/g\" {} +"
print_and_execute "find . -type f -exec sed -i \"s/YOUR_REGION/${REGION}/g\" {} +"
if [[ -n ${FOLDER_ID} ]]; then
sed -i "s/YOUR_FOLDER_ID/${FOLDER_ID}/" env/*/variables.tf
else
sed -i "s/YOUR_FOLDER_ID//" env/*/variables.tf
fi
title_no_wait "Replacing tf bucket in backend.tf in ${INFRA_SETUP_REPO}..."
sed -i "s/YOUR_PLATFORM_INFRA_TERRAFORM_STATE_BUCKET/${INFRA_TF_BUCKET}/" env/*/backend.tf
title_no_wait "Committing and pushing changes to ${INFRA_SETUP_REPO}..."
git add env/*/variables.tf env/*/backend.tf env/*/main.tf
git config --global user.name ${GITHUB_USER}
git config --global user.email "${GITHUB_USER}github.com"
git commit -m "Initial setup"
git push
title_no_wait "The push to the ${INFRA_SETUP_REPO} has started the cloudbuild trigger. Go to https://console.cloud.google.com/cloud-build/builds?project=${INFRA_SETUP_PROJECT_ID} ."
title_no_wait "Removing temp directory"
print_and_execute "rm -rf ${TEMP_DIR}"
title_no_wait "FINISHED"