forked from sourcetoad/aws-codedeploy-action
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathdeploy.sh
executable file
·248 lines (202 loc) · 8.12 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
#!/bin/bash
set -e
RESET_TEXT='\033[0m'
RED='\033[0;31m'
GREEN='\033[0;32m'
ORANGE='\033[0;33m'
BLUE='\033[0;34m'
# 0) Validation
if [ -z "$INPUT_CODEDEPLOY_NAME" ] && [ -z "$INPUT_DRY_RUN" ]; then
echo "::error::codedeploy_name is required and must not be empty."
exit 1;
fi
if [ -z "$INPUT_CODEDEPLOY_GROUP" ] && [ -z "$INPUT_DRY_RUN" ]; then
echo "::error::codedeploy_group is required and must not be empty."
exit 1;
fi
if [ -z "$INPUT_S3_BUCKET" ] && [ -z "$INPUT_DRY_RUN" ]; then
echo "::error::s3_bucket is required and must not be empty."
exit 1;
fi
echo "::debug::Input variables correctly validated."
# 0.5) Validation of AWS Creds
AWS_USERID=$(aws sts get-caller-identity | jq -r '.UserId')
if [ -z "$AWS_USERID" ] && [ -z "$INPUT_DRY_RUN" ]; then
echo "::error::Access could not be reached to AWS. Double check aws-actions/configure-aws-credentials or aws_access_key/aws_secret_key."
exit 1;
fi
# 1) Load our permissions in for aws-cli
if [ -n "$INPUT_AWS_ACCESS_KEY" ]; then
export AWS_ACCESS_KEY_ID=$INPUT_AWS_ACCESS_KEY
fi
if [ -n "$INPUT_AWS_SECRET_KEY" ]; then
export AWS_SECRET_ACCESS_KEY=$INPUT_AWS_SECRET_KEY
fi
if [ -n "$INPUT_AWS_REGION" ]; then
export AWS_DEFAULT_REGION=$INPUT_AWS_REGION
fi
# 2) Zip up the package, if no archive given
if [ -z "$INPUT_ARCHIVE" ]; then
DIR_TO_ZIP="./$INPUT_DIRECTORY"
if [ ! -f "$DIR_TO_ZIP/appspec.yml" ]; then
echo "::error::appspec.yml was not located at: $DIR_TO_ZIP"
exit 1;
fi
echo "::debug::Zip directory located (with appspec.yml)."
ZIP_FILENAME=$GITHUB_RUN_ID-$GITHUB_SHA.zip
# This creates a temp file to explode space delimited excluded files
# into newline delimited exclusions passed to "-x" on the zip command.
EXCLUSION_FILE=$(mktemp /tmp/zip-excluded.XXXXXX)
echo "$INPUT_EXCLUDED_FILES" | tr ' ' '\n' > "$EXCLUSION_FILE"
echo "::debug::Exclusion file created for files to ignore in Zip Generation."
if [ -n "$DIR_TO_ZIP" ]; then
cd "$DIR_TO_ZIP";
fi
# shellcheck disable=SC2086
zip $INPUT_CUSTOM_ZIP_FLAGS -r --quiet "$ZIP_FILENAME" . -x "@$EXCLUSION_FILE"
if [ ! -f "$ZIP_FILENAME" ]; then
echo "::error::$ZIP_FILENAME was not generated properly (zip generation failed)."
exit 1;
fi
echo "::debug::Zip Archive created."
else
echo "::debug::$INPUT_ARCHIVE being using as zip filename. Skipping generation of ZIP."
ZIP_FILENAME="$INPUT_ARCHIVE"
fi
BUNDLE_TYPE=''
# permitted values for BUNDLE_TYPE can be found here: https://docs.aws.amazon.com/codedeploy/latest/userguide/application-revisions-push.html#push-with-cli
case "$ZIP_FILENAME" in
*.tar)
BUNDLE_TYPE=tar
;;
*.tar.gz)
BUNDLE_TYPE=tgz
;;
*)
# assume it's a zipfile
BUNDLE_TYPE=zip
;;
esac
if [ "$BUNDLE_TYPE" == 'zip' ]; then
if [ "$(unzip -l "$ZIP_FILENAME" | grep -q appspec.yml)" = "0" ]; then
echo "::error::$ZIP_FILENAME was not generated properly (missing appspec.yml)."
exit 1;
fi
else
if ! tar -tf "$ZIP_FILENAME" | grep -q appspec.yml; then
echo "::error::$ZIP_FILENAME was not generated properly (missing appspec.yml)."
exit 1;
fi
fi
echo "::debug::Zip Archived validated."
echo "zip_filename=$ZIP_FILENAME" >> "$GITHUB_OUTPUT"
# 3) Upload the deployment to S3, drop old archive.
function getArchiveETag() {
aws s3api head-object \
--bucket "$INPUT_S3_BUCKET" \
--key "$INPUT_S3_FOLDER"/"$ZIP_FILENAME" \
--query ETag --output text
}
if "$INPUT_DRY_RUN"; then
echo "::debug::Dry Run detected. Exiting."
exit 0;
fi
aws s3 cp "$ZIP_FILENAME" s3://"$INPUT_S3_BUCKET"/"$INPUT_S3_FOLDER"/"$ZIP_FILENAME"
echo "::debug::Zip uploaded to S3."
ZIP_ETAG=$(getArchiveETag)
echo "::debug::Obtained ETag of uploaded S3 Zip Archive."
echo "etag=$ZIP_ETAG" >> "$GITHUB_OUTPUT"
rm "$ZIP_FILENAME"
echo "::debug::Removed old local ZIP Archive."
# 4) Start the CodeDeploy
function getActiveDeployments() {
if ! aws deploy list-deployments \
--application-name "$INPUT_CODEDEPLOY_NAME" \
--deployment-group-name "$INPUT_CODEDEPLOY_GROUP" \
--include-only-statuses "Queued" "InProgress" | jq -r '.deployments'; then
echo -e "${ORANGE}Deployment may still be executing."
echo -e "${RED}Failed monitoring deployment (ListDeployments API call failed)."
exit 1;
fi
}
function getSpecificDeployment() {
aws deploy get-deployment \
--deployment-id "$1";
}
function pollForSpecificDeployment() {
deadlockCounter=0;
while true; do
RESPONSE=$(getSpecificDeployment "$1")
FAILED_COUNT=$(echo "$RESPONSE" | jq -r '.deploymentInfo.deploymentOverview.Failed')
IN_PROGRESS_COUNT=$(echo "$RESPONSE" | jq -r '.deploymentInfo.deploymentOverview.InProgress')
SKIPPED_COUNT=$(echo "$RESPONSE" | jq -r '.deploymentInfo.deploymentOverview.Skipped')
SUCCESS_COUNT=$(echo "$RESPONSE" | jq -r '.deploymentInfo.deploymentOverview.Succeeded')
PENDING_COUNT=$(echo "$RESPONSE" | jq -r '.deploymentInfo.deploymentOverview.Pending')
STATUS=$(echo "$RESPONSE" | jq -r '.deploymentInfo.status')
echo -e "${ORANGE}Deployment in progress. Sleeping 15 seconds. (Try $((++deadlockCounter)))";
echo -e "Instance Overview: ${RED}Failed ($FAILED_COUNT), ${BLUE}In-Progress ($IN_PROGRESS_COUNT), ${RESET_TEXT}Skipped ($SKIPPED_COUNT), ${BLUE}Pending ($PENDING_COUNT), ${GREEN}Succeeded ($SUCCESS_COUNT)"
echo -e "Deployment Status: $STATUS"
if [ "$FAILED_COUNT" -gt 0 ]; then
echo -e "${RED}Failed instance detected (Failed count over zero)."
exit 1;
fi
if [ "$STATUS" = "Failed" ]; then
echo -e "${RED}Failed deployment detected (Failed status)."
exit 1;
fi
if [ "$STATUS" = "Succeeded" ]; then
break;
fi
if [ "$deadlockCounter" -gt "$INPUT_MAX_POLLING_ITERATIONS" ]; then
echo -e "${RED}Max polling iterations reached (max_polling_iterations)."
exit 1;
fi
sleep 15s;
done;
}
function pollForActiveDeployments() {
deadlockCounter=0;
while [ "$(getActiveDeployments)" != "[]" ]; do
echo -e "${ORANGE}Deployment in progress. Sleeping 15 seconds. (Try $((++deadlockCounter)))";
if [ "$deadlockCounter" -gt "$INPUT_MAX_POLLING_ITERATIONS" ]; then
echo -e "${RED}Max polling iterations reached (max_polling_iterations)."
exit 1;
fi
sleep 15s;
done;
}
pollForActiveDeployments
# 5) Poll / Complete
function deployRevision() {
aws deploy create-deployment "$@" \
--application-name "$INPUT_CODEDEPLOY_NAME" \
--deployment-group-name "$INPUT_CODEDEPLOY_GROUP" \
--description "$GITHUB_REF - $GITHUB_SHA" \
--s3-location bucket="$INPUT_S3_BUCKET",bundleType="$BUNDLE_TYPE",eTag="$ZIP_ETAG",key="$INPUT_S3_FOLDER"/"$ZIP_FILENAME" | jq -r '.deploymentId'
}
function registerRevision() {
aws deploy register-application-revision \
--application-name "$INPUT_CODEDEPLOY_NAME" \
--description "$GITHUB_REF - $GITHUB_SHA" \
--s3-location bucket="$INPUT_S3_BUCKET",bundleType="$BUNDLE_TYPE",eTag="$ZIP_ETAG",key="$INPUT_S3_FOLDER"/"$ZIP_FILENAME" > /dev/null 2>&1
}
if $INPUT_CODEDEPLOY_REGISTER_ONLY; then
echo -e "${BLUE}Registering deployment to ${RESET_TEXT}$INPUT_CODEDEPLOY_GROUP.";
registerRevision
echo -e "${BLUE}Registered deployment to ${RESET_TEXT}$INPUT_CODEDEPLOY_GROUP!";
else
echo -e "${BLUE}Deploying to ${RESET_TEXT}$INPUT_CODEDEPLOY_GROUP.";
if [ -n "$INPUT_CODEDEPLOY_CONFIG_NAME" ]; then
DEPLOYMENT_ID=$(deployRevision --deployment-config-name "$INPUT_CODEDEPLOY_CONFIG_NAME")
else
DEPLOYMENT_ID=$(deployRevision)
fi
if [ "$INPUT_MAX_POLLING_ITERATIONS" -eq "0" ]; then
echo -e "${BLUE}Iterations at 0. GitHub Action ending, but deployment in-progress to: ${RESET_TEXT}$INPUT_CODEDEPLOY_GROUP.";
else
sleep 10;
pollForSpecificDeployment "$DEPLOYMENT_ID"
echo -e "${GREEN}Deployed to ${RESET_TEXT}$INPUT_CODEDEPLOY_GROUP!";
fi
fi
exit 0;