2018-12-05 23:21:45 +05:30
|
|
|
[[ "$TRACE" ]] && set -x
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
function namespace_exists() {
|
|
|
|
local namespace="${1}"
|
|
|
|
local namespace_exists
|
|
|
|
|
|
|
|
echoinfo "Checking if ${namespace} exists..." true
|
|
|
|
|
|
|
|
kubectl describe namespace "${namespace}" >/dev/null 2>&1
|
|
|
|
namespace_exists=$?
|
|
|
|
|
|
|
|
if [ $namespace_exists -eq 0 ]; then
|
|
|
|
echoinfo "Namespace ${namespace} found."
|
|
|
|
else
|
|
|
|
echoerr "Namespace ${namespace} NOT found."
|
|
|
|
fi
|
|
|
|
|
|
|
|
return $namespace_exists
|
|
|
|
}
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
function deploy_exists() {
|
2019-07-07 11:18:12 +05:30
|
|
|
local namespace="${1}"
|
2020-01-01 13:55:28 +05:30
|
|
|
local release="${2}"
|
|
|
|
local deploy_exists
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2020-01-01 13:55:28 +05:30
|
|
|
echoinfo "Checking if ${release} exists in the ${namespace} namespace..." true
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
helm status --namespace "${namespace}" "${release}" >/dev/null 2>&1
|
2020-01-01 13:55:28 +05:30
|
|
|
deploy_exists=$?
|
|
|
|
|
2021-12-11 22:18:48 +05:30
|
|
|
if [ $deploy_exists -eq 0 ]; then
|
|
|
|
echoinfo "Previous deployment for ${release} found."
|
|
|
|
else
|
|
|
|
echoerr "Previous deployment for ${release} NOT found."
|
|
|
|
fi
|
|
|
|
|
2019-07-07 11:18:12 +05:30
|
|
|
return $deploy_exists
|
|
|
|
}
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
function previous_deploy_failed() {
|
2019-12-26 22:10:19 +05:30
|
|
|
local namespace="${1}"
|
2020-01-01 13:55:28 +05:30
|
|
|
local release="${2}"
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-01-01 13:55:28 +05:30
|
|
|
echoinfo "Checking for previous deployment of ${release}" true
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
helm status --namespace "${namespace}" "${release}" >/dev/null 2>&1
|
2019-07-07 11:18:12 +05:30
|
|
|
local status=$?
|
|
|
|
|
|
|
|
# if `status` is `0`, deployment exists, has a status
|
|
|
|
if [ $status -eq 0 ]; then
|
|
|
|
echoinfo "Previous deployment found, checking status..."
|
2020-04-22 19:07:51 +05:30
|
|
|
deployment_status=$(helm status --namespace "${namespace}" "${release}" | grep ^STATUS | cut -d' ' -f2)
|
2019-07-07 11:18:12 +05:30
|
|
|
echoinfo "Previous deployment state: ${deployment_status}"
|
2020-04-22 19:07:51 +05:30
|
|
|
if [[ "$deployment_status" == "failed" || "$deployment_status" == "pending-upgrade" || "$deployment_status" == "pending-install" ]]; then
|
2019-07-07 11:18:12 +05:30
|
|
|
status=0;
|
|
|
|
else
|
|
|
|
status=1;
|
|
|
|
fi
|
|
|
|
else
|
|
|
|
echoerr "Previous deployment NOT found."
|
|
|
|
fi
|
|
|
|
return $status
|
|
|
|
}
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
function delete_helm_release() {
|
2021-09-04 01:27:46 +05:30
|
|
|
local namespace="${CI_ENVIRONMENT_SLUG}"
|
2020-01-01 13:55:28 +05:30
|
|
|
local release="${CI_ENVIRONMENT_SLUG}"
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-01-01 13:55:28 +05:30
|
|
|
if [ -z "${release}" ]; then
|
2019-07-07 11:18:12 +05:30
|
|
|
echoerr "No release given, aborting the delete!"
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
|
2021-09-30 23:02:18 +05:30
|
|
|
if deploy_exists "${namespace}" "${release}"; then
|
|
|
|
helm uninstall --namespace="${namespace}" "${release}"
|
|
|
|
fi
|
2021-09-04 01:27:46 +05:30
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
if namespace_exists "${namespace}"; then
|
|
|
|
echoinfo "Deleting namespace ${namespace}..." true
|
|
|
|
kubectl delete namespace "${namespace}" --wait
|
|
|
|
fi
|
2021-09-04 01:27:46 +05:30
|
|
|
}
|
|
|
|
|
2019-07-07 11:18:12 +05:30
|
|
|
function get_pod() {
|
2021-09-04 01:27:46 +05:30
|
|
|
local namespace="${CI_ENVIRONMENT_SLUG}"
|
2020-01-01 13:55:28 +05:30
|
|
|
local release="${CI_ENVIRONMENT_SLUG}"
|
2019-07-07 11:18:12 +05:30
|
|
|
local app_name="${1}"
|
|
|
|
local status="${2-Running}"
|
2020-01-01 13:55:28 +05:30
|
|
|
|
|
|
|
get_pod_cmd="kubectl get pods --namespace ${namespace} --field-selector=status.phase=${status} -lapp=${app_name},release=${release} --no-headers -o=custom-columns=NAME:.metadata.name | tail -n 1"
|
2019-09-30 21:07:59 +05:30
|
|
|
echoinfo "Waiting till '${app_name}' pod is ready" true
|
|
|
|
echoinfo "Running '${get_pod_cmd}'"
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
local interval=5
|
|
|
|
local elapsed_seconds=0
|
|
|
|
local max_seconds=$((2 * 60))
|
2019-07-07 11:18:12 +05:30
|
|
|
while true; do
|
2019-07-31 22:56:46 +05:30
|
|
|
local pod_name
|
|
|
|
pod_name="$(eval "${get_pod_cmd}")"
|
2019-07-07 11:18:12 +05:30
|
|
|
[[ "${pod_name}" == "" ]] || break
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
if [[ "${elapsed_seconds}" -gt "${max_seconds}" ]]; then
|
|
|
|
echoerr "The pod name couldn't be found after ${elapsed_seconds} seconds, aborting."
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
|
|
|
|
let "elapsed_seconds+=interval"
|
|
|
|
sleep ${interval}
|
2019-07-07 11:18:12 +05:30
|
|
|
done
|
|
|
|
|
|
|
|
echoinfo "The pod name is '${pod_name}'."
|
|
|
|
echo "${pod_name}"
|
|
|
|
}
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
function run_task() {
|
2021-09-04 01:27:46 +05:30
|
|
|
local namespace="${CI_ENVIRONMENT_SLUG}"
|
2020-07-28 23:09:34 +05:30
|
|
|
local ruby_cmd="${1}"
|
2022-01-26 12:08:38 +05:30
|
|
|
local toolbox_pod=$(get_pod "toolbox")
|
2020-07-28 23:09:34 +05:30
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
run_timed_command "kubectl exec --namespace \"${namespace}\" \"${toolbox_pod}\" -- gitlab-rails runner \"${ruby_cmd}\""
|
2020-07-28 23:09:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
function disable_sign_ups() {
|
|
|
|
if [ -z ${REVIEW_APPS_ROOT_TOKEN+x} ]; then
|
|
|
|
echoerr "In order to protect Review Apps, REVIEW_APPS_ROOT_TOKEN variable must be set"
|
|
|
|
false
|
|
|
|
else
|
|
|
|
true
|
|
|
|
fi
|
|
|
|
|
2023-07-09 08:55:56 +05:30
|
|
|
# Create the root token + Disable sign-ups
|
|
|
|
#
|
|
|
|
# We use this weird syntax because we need to pass a one-liner ruby command to a Kubernetes container via kubectl.
|
|
|
|
read -r -d '' multiline_ruby_code <<RUBY
|
|
|
|
user = User.find_by_username('root');
|
|
|
|
puts 'Error: Could not find root user. Check that the database was properly seeded'; exit(1) unless user;
|
|
|
|
token = user.personal_access_tokens.create(scopes: [:api], name: 'Token to disable sign-ups');
|
|
|
|
token.set_token('${REVIEW_APPS_ROOT_TOKEN}');
|
|
|
|
begin;
|
|
|
|
token.save!;
|
|
|
|
rescue(ActiveRecord::RecordNotUnique);
|
|
|
|
end;
|
|
|
|
Gitlab::CurrentSettings.current_application_settings.update!(signup_enabled: false);
|
|
|
|
RUBY
|
|
|
|
|
|
|
|
local disable_signup_rb=$(echo $multiline_ruby_code | tr '\n' ' ')
|
2023-04-23 21:23:45 +05:30
|
|
|
if (retry_exponential "run_task \"${disable_signup_rb}\""); then
|
2020-07-28 23:09:34 +05:30
|
|
|
echoinfo "Sign-ups have been disabled successfully."
|
|
|
|
else
|
2020-11-24 15:15:51 +05:30
|
|
|
echoerr "Sign-ups are still enabled!"
|
2020-07-28 23:09:34 +05:30
|
|
|
false
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
function create_sample_projects() {
|
|
|
|
local create_sample_projects_rb="root_user = User.find_by_username('root'); 1.times { |i| params = { namespace_id: root_user.namespace.id, name: 'sample-project' + i.to_s, path: 'sample-project' + i.to_s, template_name: 'sample' }; ::Projects::CreateFromTemplateService.new(root_user, params).execute }"
|
|
|
|
|
|
|
|
# Queue jobs to create sample projects for root user namespace from sample data project template
|
|
|
|
retry "run_task \"${create_sample_projects_rb}\""
|
|
|
|
}
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
function check_kube_domain() {
|
2019-07-07 11:18:12 +05:30
|
|
|
echoinfo "Checking that Kube domain exists..." true
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
if [ -z ${REVIEW_APPS_DOMAIN+x} ]; then
|
|
|
|
echo "In order to deploy or use Review Apps, REVIEW_APPS_DOMAIN variable must be set"
|
|
|
|
echo "You can do it in Auto DevOps project settings or defining a variable at group or project level"
|
|
|
|
echo "You can also manually add it in .gitlab-ci.yml"
|
|
|
|
false
|
|
|
|
else
|
|
|
|
true
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2019-05-30 16:15:17 +05:30
|
|
|
function ensure_namespace() {
|
2021-09-04 01:27:46 +05:30
|
|
|
local namespace="${1}"
|
2020-01-01 13:55:28 +05:30
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
if ! namespace_exists "${namespace}"; then
|
|
|
|
echoinfo "Creating namespace ${namespace}..." true
|
|
|
|
kubectl create namespace "${namespace}"
|
|
|
|
fi
|
2018-12-05 23:21:45 +05:30
|
|
|
}
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
function label_namespace() {
|
|
|
|
local namespace="${1}"
|
|
|
|
local label="${2}"
|
|
|
|
|
|
|
|
echoinfo "Labeling the ${namespace} namespace with ${label}" true
|
2021-09-30 23:02:18 +05:30
|
|
|
echoinfo "We should pass the --overwrite option!"
|
2021-09-04 01:27:46 +05:30
|
|
|
|
2021-09-30 23:02:18 +05:30
|
|
|
kubectl label --overwrite namespace "${namespace}" "${label}"
|
2020-07-28 23:09:34 +05:30
|
|
|
}
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
function create_application_secret() {
|
2021-09-04 01:27:46 +05:30
|
|
|
local namespace="${CI_ENVIRONMENT_SLUG}"
|
2020-01-01 13:55:28 +05:30
|
|
|
local release="${CI_ENVIRONMENT_SLUG}"
|
|
|
|
local initial_root_password_shared_secret
|
|
|
|
local gitlab_license_shared_secret
|
|
|
|
|
2022-06-21 17:19:12 +05:30
|
|
|
initial_root_password_shared_secret=$(kubectl get secret --namespace ${namespace} --no-headers -o=custom-columns=NAME:.metadata.name shared-gitlab-initial-root-password 2> /dev/null | tail -n 1)
|
2020-01-01 13:55:28 +05:30
|
|
|
if [[ "${initial_root_password_shared_secret}" == "" ]]; then
|
|
|
|
echoinfo "Creating the 'shared-gitlab-initial-root-password' secret in the ${namespace} namespace..." true
|
|
|
|
kubectl create secret generic --namespace "${namespace}" \
|
|
|
|
"shared-gitlab-initial-root-password" \
|
|
|
|
--from-literal="password=${REVIEW_APPS_ROOT_PASSWORD}" \
|
2022-08-27 11:52:29 +05:30
|
|
|
--dry-run=client -o json | kubectl apply -f -
|
2020-01-01 13:55:28 +05:30
|
|
|
else
|
|
|
|
echoinfo "The 'shared-gitlab-initial-root-password' secret already exists in the ${namespace} namespace."
|
|
|
|
fi
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2023-07-09 08:55:56 +05:30
|
|
|
if [ -z "${QA_EE_LICENSE}" ]; then echo "License not found" && return; fi
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
gitlab_license_shared_secret=$(kubectl get secret --namespace "${namespace}" --no-headers -o=custom-columns=NAME:.metadata.name shared-gitlab-license 2> /dev/null | tail -n 1)
|
2020-01-01 13:55:28 +05:30
|
|
|
if [[ "${gitlab_license_shared_secret}" == "" ]]; then
|
2023-03-04 22:38:38 +05:30
|
|
|
echoinfo "Creating the 'shared-gitlab-license' secret in the "${namespace}" namespace..." true
|
2020-01-01 13:55:28 +05:30
|
|
|
kubectl create secret generic --namespace "${namespace}" \
|
|
|
|
"shared-gitlab-license" \
|
2023-07-09 08:55:56 +05:30
|
|
|
--from-literal=license="${QA_EE_LICENSE}" \
|
2022-08-27 11:52:29 +05:30
|
|
|
--dry-run=client -o json | kubectl apply -f -
|
2020-01-01 13:55:28 +05:30
|
|
|
else
|
|
|
|
echoinfo "The 'shared-gitlab-license' secret already exists in the ${namespace} namespace."
|
|
|
|
fi
|
2018-12-05 23:21:45 +05:30
|
|
|
}
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
function download_chart() {
|
2022-11-25 23:54:43 +05:30
|
|
|
# If the requirements.lock is present, it means we got everything we need from the cache.
|
|
|
|
if [[ -f "gitlab-${GITLAB_HELM_CHART_REF}/requirements.lock" ]]; then
|
|
|
|
echosuccess "Downloading/Building chart dependencies skipped. Using the chart ${gitlab-${GITLAB_HELM_CHART_REF}} local folder'..."
|
|
|
|
else
|
|
|
|
echoinfo "Downloading the GitLab chart..." true
|
2019-05-18 00:54:41 +05:30
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
curl --location -o gitlab.tar.bz2 "https://gitlab.com/gitlab-org/charts/gitlab/-/archive/${GITLAB_HELM_CHART_REF}/gitlab-${GITLAB_HELM_CHART_REF}.tar.bz2"
|
|
|
|
tar -xjf gitlab.tar.bz2
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
echoinfo "Adding the gitlab repo to Helm..."
|
|
|
|
helm repo add gitlab https://charts.gitlab.io
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
echoinfo "Building the gitlab chart's dependencies..."
|
|
|
|
helm dependency build "gitlab-${GITLAB_HELM_CHART_REF}"
|
|
|
|
fi
|
2018-12-05 23:21:45 +05:30
|
|
|
}
|
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
function base_config_changed() {
|
|
|
|
if [ -z "${CI_MERGE_REQUEST_IID}" ]; then return; fi
|
|
|
|
|
|
|
|
curl "${CI_API_V4_URL}/projects/${CI_MERGE_REQUEST_PROJECT_ID}/merge_requests/${CI_MERGE_REQUEST_IID}/changes" | jq '.changes | any(.old_path == "scripts/review_apps/base-config.yaml")'
|
|
|
|
}
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
function parse_gitaly_image_tag() {
|
|
|
|
local gitaly_version="${GITALY_VERSION}"
|
|
|
|
|
|
|
|
# prepend semver version with `v`
|
|
|
|
if [[ $gitaly_version =~ ^[0-9]+\.[0-9]+\.[0-9]+(-rc[0-9]+)?(-ee)?$ ]]; then
|
|
|
|
echo "v${gitaly_version}"
|
|
|
|
else
|
|
|
|
echo "${gitaly_version}"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
function deploy() {
|
2021-09-04 01:27:46 +05:30
|
|
|
local namespace="${CI_ENVIRONMENT_SLUG}"
|
2020-01-01 13:55:28 +05:30
|
|
|
local release="${CI_ENVIRONMENT_SLUG}"
|
2021-06-08 01:23:25 +05:30
|
|
|
local base_config_file_ref="${CI_DEFAULT_BRANCH}"
|
2020-01-01 13:55:28 +05:30
|
|
|
if [[ "$(base_config_changed)" == "true" ]]; then base_config_file_ref="${CI_COMMIT_SHA}"; fi
|
2019-12-26 22:10:19 +05:30
|
|
|
local base_config_file="https://gitlab.com/gitlab-org/gitlab/raw/${base_config_file_ref}/scripts/review_apps/base-config.yaml"
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
echoinfo "Deploying ${release} to ${CI_ENVIRONMENT_URL} ..." true
|
2018-12-05 23:21:45 +05:30
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
IMAGE_REPOSITORY="registry.gitlab.com/gitlab-org/build/cng-mirror"
|
2021-11-11 11:23:49 +05:30
|
|
|
gitlab_toolbox_image_repository="${IMAGE_REPOSITORY}/gitlab-toolbox-ee"
|
2020-05-24 23:13:21 +05:30
|
|
|
gitlab_sidekiq_image_repository="${IMAGE_REPOSITORY}/gitlab-sidekiq-ee"
|
2020-06-23 00:09:42 +05:30
|
|
|
gitlab_webservice_image_repository="${IMAGE_REPOSITORY}/gitlab-webservice-ee"
|
2019-02-15 15:39:39 +05:30
|
|
|
gitlab_gitaly_image_repository="${IMAGE_REPOSITORY}/gitaly"
|
2020-07-28 23:09:34 +05:30
|
|
|
gitaly_image_tag=$(parse_gitaly_image_tag)
|
2019-02-15 15:39:39 +05:30
|
|
|
gitlab_shell_image_repository="${IMAGE_REPOSITORY}/gitlab-shell"
|
2020-05-24 23:13:21 +05:30
|
|
|
gitlab_workhorse_image_repository="${IMAGE_REPOSITORY}/gitlab-workhorse-ee"
|
2022-01-26 12:08:38 +05:30
|
|
|
sentry_enabled="false"
|
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
if [ -n "${REVIEW_APPS_SENTRY_DSN}" ]; then
|
2022-01-26 12:08:38 +05:30
|
|
|
echo "REVIEW_APPS_SENTRY_DSN detected, enabling Sentry"
|
|
|
|
sentry_enabled="true"
|
|
|
|
fi
|
2018-12-05 23:21:45 +05:30
|
|
|
|
2023-04-23 21:23:45 +05:30
|
|
|
retry "ensure_namespace \"${namespace}\""
|
|
|
|
retry "label_namespace \"${namespace}\" \"tls=review-apps-tls\"" # label namespace for kubed to sync tls
|
2021-09-04 01:27:46 +05:30
|
|
|
|
2023-04-23 21:23:45 +05:30
|
|
|
retry "create_application_secret"
|
2018-12-05 23:21:45 +05:30
|
|
|
|
2022-07-23 23:45:48 +05:30
|
|
|
cat > review_apps.values.yml <<EOF
|
2023-06-20 00:43:36 +05:30
|
|
|
ci:
|
|
|
|
branch: "${CI_COMMIT_REF_NAME}"
|
|
|
|
commit:
|
|
|
|
sha: "${CI_COMMIT_SHORT_SHA}"
|
|
|
|
job:
|
|
|
|
url: "${CI_JOB_URL}"
|
|
|
|
pipeline:
|
|
|
|
url: "${CI_PIPELINE_URL}"
|
|
|
|
|
2022-07-23 23:45:48 +05:30
|
|
|
gitlab:
|
2023-06-20 00:43:36 +05:30
|
|
|
gitaly:
|
|
|
|
image:
|
|
|
|
repository: "${gitlab_gitaly_image_repository}"
|
|
|
|
tag: "${gitaly_image_tag}"
|
|
|
|
gitlab-shell:
|
|
|
|
image:
|
|
|
|
repository: "${gitlab_shell_image_repository}"
|
|
|
|
tag: "v${GITLAB_SHELL_VERSION}"
|
|
|
|
migrations:
|
|
|
|
image:
|
|
|
|
repository: "${gitlab_toolbox_image_repository}"
|
|
|
|
tag: "${CI_COMMIT_SHA}"
|
|
|
|
sidekiq:
|
|
|
|
annotations:
|
|
|
|
commit: "${CI_COMMIT_SHORT_SHA}"
|
|
|
|
image:
|
|
|
|
repository: "${gitlab_sidekiq_image_repository}"
|
|
|
|
tag: "${CI_COMMIT_SHA}"
|
|
|
|
toolbox:
|
|
|
|
image:
|
|
|
|
repository: "${gitlab_toolbox_image_repository}"
|
|
|
|
tag: "${CI_COMMIT_SHA}"
|
2022-07-23 23:45:48 +05:30
|
|
|
webservice:
|
2023-06-20 00:43:36 +05:30
|
|
|
annotations:
|
|
|
|
commit: "${CI_COMMIT_SHORT_SHA}"
|
2022-07-23 23:45:48 +05:30
|
|
|
extraEnv:
|
|
|
|
REVIEW_APPS_ENABLED: "true"
|
|
|
|
REVIEW_APPS_MERGE_REQUEST_IID: "${CI_MERGE_REQUEST_IID}"
|
2023-06-20 00:43:36 +05:30
|
|
|
image:
|
|
|
|
repository: "${gitlab_webservice_image_repository}"
|
|
|
|
tag: "${CI_COMMIT_SHA}"
|
|
|
|
workhorse:
|
|
|
|
image: "${gitlab_workhorse_image_repository}"
|
|
|
|
tag: "${CI_COMMIT_SHA}"
|
|
|
|
|
|
|
|
global:
|
|
|
|
hosts:
|
|
|
|
domain: "${REVIEW_APPS_DOMAIN}"
|
|
|
|
hostSuffix: "${HOST_SUFFIX}"
|
|
|
|
appConfig:
|
|
|
|
sentry:
|
|
|
|
dsn: "${REVIEW_APPS_SENTRY_DSN}"
|
|
|
|
# Boolean fields should be left without quotes
|
|
|
|
enabled: ${sentry_enabled}
|
|
|
|
environment: "review"
|
|
|
|
|
|
|
|
releaseOverride: "${release}"
|
2022-07-23 23:45:48 +05:30
|
|
|
EOF
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
HELM_CMD=$(cat << EOF
|
2020-01-01 13:55:28 +05:30
|
|
|
helm upgrade \
|
|
|
|
--namespace="${namespace}" \
|
2021-09-04 01:27:46 +05:30
|
|
|
--create-namespace \
|
2020-01-01 13:55:28 +05:30
|
|
|
--install \
|
2019-12-04 20:38:33 +05:30
|
|
|
--wait \
|
2023-06-20 00:43:36 +05:30
|
|
|
--timeout "${HELM_INSTALL_TIMEOUT:-20m}"
|
2019-09-30 21:07:59 +05:30
|
|
|
EOF
|
|
|
|
)
|
|
|
|
|
2023-07-09 08:55:56 +05:30
|
|
|
if [ -n "${QA_EE_LICENSE}" ]; then
|
2019-12-26 22:10:19 +05:30
|
|
|
HELM_CMD=$(cat << EOF
|
|
|
|
${HELM_CMD} \
|
2023-01-13 00:05:48 +05:30
|
|
|
--set global.gitlab.license.secret="shared-gitlab-license"
|
2019-12-26 22:10:19 +05:30
|
|
|
EOF
|
|
|
|
)
|
|
|
|
fi
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
# Important: the `-f` calls are ordered. They should not be changed.
|
|
|
|
#
|
|
|
|
# The `base_config_file` contains the default values for the chart, and the
|
|
|
|
# `review_apps.values.yml` contains the overrides we want to apply specifically
|
|
|
|
# for this review app deployment.
|
2019-09-30 21:07:59 +05:30
|
|
|
HELM_CMD=$(cat << EOF
|
2019-12-26 22:10:19 +05:30
|
|
|
${HELM_CMD} \
|
2023-01-13 00:05:48 +05:30
|
|
|
--version="${CI_PIPELINE_ID}-${CI_JOB_ID}" \
|
|
|
|
-f "${base_config_file}" \
|
2023-06-20 00:43:36 +05:30
|
|
|
-f review_apps.values.yml \
|
2023-01-13 00:05:48 +05:30
|
|
|
-v "${HELM_LOG_VERBOSITY:-1}" \
|
|
|
|
"${release}" "gitlab-${GITLAB_HELM_CHART_REF}"
|
2018-12-05 23:21:45 +05:30
|
|
|
EOF
|
|
|
|
)
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
# Pretty-print the command for display
|
2023-06-20 00:43:36 +05:30
|
|
|
echoinfo "Deploying with helm command:"
|
2023-01-13 00:05:48 +05:30
|
|
|
echo "${HELM_CMD}" | sed 's/ /\n\t/g'
|
2018-12-05 23:21:45 +05:30
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
echoinfo "Content of review_apps.values.yml:"
|
|
|
|
cat review_apps.values.yml
|
|
|
|
|
2023-04-23 21:23:45 +05:30
|
|
|
retry "eval \"${HELM_CMD}\""
|
2019-09-30 21:07:59 +05:30
|
|
|
}
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
function verify_deploy() {
|
2023-03-04 22:38:38 +05:30
|
|
|
local deployed="false"
|
|
|
|
|
|
|
|
mkdir -p curl-logs/
|
2023-01-13 00:05:48 +05:30
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
for i in {1..60}; do # try for 5 minutes
|
|
|
|
local now=$(date '+%H:%M:%S')
|
|
|
|
echo "[${now}] Verifying deployment at ${CI_ENVIRONMENT_URL}/users/sign_in"
|
|
|
|
log_name="curl-logs/${now}.log"
|
|
|
|
curl --connect-timeout 3 -o "${log_name}" -s "${CI_ENVIRONMENT_URL}/users/sign_in"
|
|
|
|
|
|
|
|
if grep "Remember me" "${log_name}" &> /dev/null; then
|
|
|
|
deployed="true"
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
|
|
|
|
sleep 5
|
|
|
|
done
|
2021-09-04 01:27:46 +05:30
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
if [[ "${deployed}" == "true" ]]; then
|
2023-01-13 00:05:48 +05:30
|
|
|
echoinfo "[$(date '+%H:%M:%S')] Review app is deployed to ${CI_ENVIRONMENT_URL}"
|
2021-09-04 01:27:46 +05:30
|
|
|
else
|
2023-01-13 00:05:48 +05:30
|
|
|
echoerr "[$(date '+%H:%M:%S')] Review app is not available at ${CI_ENVIRONMENT_URL}: see the logs from cURL above for more details"
|
2021-09-04 01:27:46 +05:30
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
# We need to be able to access the GitLab API to run this method.
|
|
|
|
# Since we are creating a personal access token in `disable_sign_ups`,
|
|
|
|
# This method should be executed after it.
|
|
|
|
function verify_commit_sha() {
|
|
|
|
local verify_success="false"
|
|
|
|
|
|
|
|
for i in {1..60}; do # try for 2 minutes in case review-apps containers are restarting
|
|
|
|
echoinfo "[$(date '+%H:%M:%S')] Checking the correct commit is deployed in the review-app:"
|
|
|
|
echo "Expected commit sha: ${CI_COMMIT_SHA}"
|
|
|
|
|
|
|
|
review_app_revision=$(curl --header "PRIVATE-TOKEN: ${REVIEW_APPS_ROOT_TOKEN}" "${CI_ENVIRONMENT_URL}/api/v4/metadata" | jq -r .revision)
|
|
|
|
echo "review-app revision: ${review_app_revision}"
|
|
|
|
|
|
|
|
if [[ "${CI_COMMIT_SHA}" == "${review_app_revision}"* ]]; then
|
|
|
|
verify_success="true"
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
|
|
|
|
sleep 2
|
|
|
|
done
|
|
|
|
|
|
|
|
if [[ "${verify_success}" != "true" ]]; then
|
|
|
|
echoerr "[$(date '+%H:%M:%S')] Review app revision is not the same as the current commit!"
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
function display_deployment_debug() {
|
2021-09-04 01:27:46 +05:30
|
|
|
local namespace="${CI_ENVIRONMENT_SLUG}"
|
2019-12-04 20:38:33 +05:30
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
# Install dig to inspect DNS entries
|
2023-03-17 16:20:25 +05:30
|
|
|
apk add -q bind-tools
|
2023-03-04 22:38:38 +05:30
|
|
|
|
|
|
|
echoinfo "[debugging data] Check review-app webservice DNS entry:"
|
|
|
|
dig +short $(echo "${CI_ENVIRONMENT_URL}" | sed 's~http[s]*://~~g')
|
|
|
|
|
|
|
|
echoinfo "[debugging data] Check external IP for nginx-ingress-controller service (should be THE SAME AS the DNS entry IP above):"
|
|
|
|
kubectl -n "${namespace}" get svc "${namespace}-nginx-ingress-controller" -o jsonpath='{.status.loadBalancer.ingress[].ip}'
|
|
|
|
|
|
|
|
echoinfo "[debugging data] k8s resources:"
|
|
|
|
kubectl -n "${namespace}" get pods
|
|
|
|
|
|
|
|
echoinfo "[debugging data] PostgreSQL logs:"
|
|
|
|
kubectl -n "${namespace}" logs -l app=postgresql --all-containers
|
|
|
|
|
|
|
|
echoinfo "[debugging data] DB migrations logs:"
|
|
|
|
kubectl -n "${namespace}" logs -l app=migrations --all-containers
|
|
|
|
|
|
|
|
echoinfo "[debugging data] Webservice logs:"
|
|
|
|
kubectl -n "${namespace}" logs -l app=webservice -c webservice
|
2019-07-31 22:56:46 +05:30
|
|
|
}
|