+ base64 -d + printf %s #!/usr/bin/env bash
#
# Tests if the ArgoCD instance is available on the cluster by logging in.
#
# Uses the ArgoCD session, created by previously running "argocd login", to
# generate an account token. The information is then stored in a kubernetes
# secret.
#
shopt -s inherit_errexit
set -o errexit
set -o errtrace
set -o nounset
set -o pipefail

usage() {
    echo "
Usage:
    ${0##*/} [options] COMMAND

Commands:
    generate
        Generate the API token
    login
		Test login to the ArgoCD instance.
    store
        Store the API token and relevant information
        in the integration secret.
Optional arguments:
    -d, --debug
        Activate tracing/debug mode.
    -h, --help
        Display this message.

Example:
    ${0##*/} login
" >&2
}

parse_args() {
    NAMESPACE="${NAMESPACE:-tssc}"
    while [[ $# -gt 0 ]]; do
        case "$1" in
        generate|login|store)
            SUBCOMMAND="$1"
            ;;
        -d | --debug)
            set -x
            DEBUG="--debug"
            export DEBUG
            info "Running script as: $(id)"
            ;;
        -h | --help)
            usage
            exit 0
            ;;
        *)
            fail "Unsupported argument: '$1'."
            ;;
        esac
        shift
    done
    if [[ -z "${SUBCOMMAND:-}" ]]; then
        fail "Missing subcommand."
    fi
}

fail() {
    echo "# [ERROR] ${*}" >&2
    exit 1
}

info() {
    echo "# [INFO] ${*}"
}

#
# Functions
#

# Asserts the required environment variables.
assert_variables() {
    # ArgoCD hostname (FQDN) to test.
    declare -r ARGOCD_HOSTNAME="${ARGOCD_HOSTNAME:-}"
    # ArgoCD username to use for login.
    declare -r ARGOCD_USER="${ARGOCD_USER:-admin}"
    # ArgoCD password to use for login.
    declare -r ARGOCD_PASSWORD="${ARGOCD_PASSWORD:-}"
    # Environment file to store the ArgoCD credentials.
    declare -r ARGOCD_ENV_FILE="${ARGOCD_ENV_FILE:-/tssc/argocd/env}"
    # Target secret name, to be created with ArgoCD credentials.
    declare -r SECRET_NAME="${SECRET_NAME:-tssc-argocd-integration}"
    # Secret's namespace.
    declare -r NAMESPACE="${NAMESPACE:-}"

    case "${SUBCOMMAND}" in
        login | generate)
            [[ -z "${ARGOCD_HOSTNAME}" ]] &&
                fail "ARGOCD_HOSTNAME is not set!"
            [[ -z "${ARGOCD_USER}" ]] &&
                fail "ARGOCD_USER is not set!"
            [[ -z "${ARGOCD_PASSWORD}" ]] &&
                fail "ARGOCD_PASSWORD is not set!"
            ;;
        store)
            [[ -z "${NAMESPACE}" ]] &&
                fail "NAMESPACE is not set!"
            [[ -z "${SECRET_NAME}" ]] &&
                fail "SECRET_NAME is not set!"
            ;;
        *)
            fail "Invalid subcommand provided: '${SUBCOMMAND}'."
            ;;
    esac
    info "# All environment variables are set"
}

# Executes the ArgoCD login command.
argocd_login() {
    argocd login "${ARGOCD_HOSTNAME}" \
        --grpc-web \
        --insecure \
        --skip-test-tls \
        --http-retry-max="5" \
        --username="${ARGOCD_USER}" \
        --password="${ARGOCD_PASSWORD}"
}

# Retries a few times until the ArgoCD instance is available.
test_argocd_login() {
    info "# Logging into ArgoCD on '${ARGOCD_HOSTNAME}'..."
    for i in {1..30}; do
        wait=$((i * 5))
        echo "### [${i}/30] Waiting for ${wait} seconds before retrying..."
        sleep ${wait}

        echo "# [${i}/30] Testing ArgoCD login on '${ARGOCD_HOSTNAME}'..."
        if argocd_login; then
            info "# ArgoCD is available: '${ARGOCD_HOSTNAME}'"
            return 0
        fi
    done
    fail "Could not log into ArgoCD."
}

# Generates the ArgoCD API token.
argocd_generate_token() {
    info "# Generating ArgoCD API token on '${ARGOCD_HOSTNAME}'..."
    ARGOCD_API_TOKEN="$(
        argocd account generate-token \
            --grpc-web \
            --insecure \
            --http-retry-max="5" \
            --account="${ARGOCD_USER}"
    )" || fail "ArgoCD API token could not be generated!"
    if [[ "${?}" -ne 0 || -z "${ARGOCD_API_TOKEN}" ]]; then
        fail "ArgoCD API token could not be generated!"
    fi

    info "# Storing ArgoCD API credentials in '${ARGOCD_ENV_FILE}'..."
    cat <<EOF >"${ARGOCD_ENV_FILE}" || fail "Fail to write '${ARGOCD_ENV_FILE}'!"
ARGOCD_HOSTNAME=${ARGOCD_HOSTNAME}
ARGOCD_USER=${ARGOCD_USER}
ARGOCD_PASSWORD=${ARGOCD_PASSWORD}
ARGOCD_API_TOKEN=${ARGOCD_API_TOKEN}
EOF

    info "# ArgoCD API token generated successfully!"
}

# Waits for the environment file to be available.
wait_for_env_file() {
    info "# Waiting for '${ARGOCD_ENV_FILE}' to be available..."
    for i in {1..30}; do
        wait=$((i * 5))
        echo "### [${i}/30] Waiting for '${ARGOCD_ENV_FILE}' to be available..."
        sleep ${wait}

        if [[ -r "${ARGOCD_ENV_FILE}" ]]; then
            info "# '${ARGOCD_ENV_FILE}' found and readable."
            return 0
        fi
    done
    fail "ARGOCD_ENV_FILE='${ARGOCD_ENV_FILE}' not found or not readable!"
}

# Stores the ArgoCD credentials in a Kubernetes secret.
argocd_store_credentials() {
    # Using the dry-run flag to generate the secret payload, and later on "kubectl
    # apply" to create, or update, the secret payload in the cluster.
    info "# Creating secret '${SECRET_NAME}' in namespace '${NAMESPACE}' from '${ARGOCD_ENV_FILE}'..."
    if ! (
        kubectl create secret generic "${SECRET_NAME}" \
            --namespace="${NAMESPACE}" \
            --from-env-file="${ARGOCD_ENV_FILE}" \
            --dry-run="client" \
            --output="yaml" |
            kubectl apply -f -
    ); then
        fail "Secret '${SECRET_NAME}' could not be created."
    fi
    info "# ArgoCD API credentials stored successfully."
}

#
# Main
#
main() {
    parse_args "$@"

    assert_variables

    case "${SUBCOMMAND}" in
    login)
        test_argocd_login
        ;;
    generate)
        test_argocd_login
        argocd_generate_token
        ;;
    store)
        wait_for_env_file
        argocd_store_credentials
        ;;
    *)
        fail "Invalid subcommand provided: '${SUBCOMMAND}'. " \
            "Use 'login', 'generate' or 'store'!"
        ;;
    esac
}

if [ "${BASH_SOURCE[0]}" == "$0" ]; then
    main "$@"
    echo
    echo "Success"
fi
 + chmod +x argocd-helper.sh + printf %s IyEvdXNyL2Jpbi9lbnYgYmFzaAojCiMgUnVucyAib2Mgcm9sbG91dCBzdGF0dXMiIGZvciBjb25maWd1cmVkIG5hbWVzcGFjZSwgcmVzb3VyY2UgdHlwZSwgYW5kIHNlbGVjdG9ycy4KIwpzaG9wdCAtcyBpbmhlcml0X2VycmV4aXQKc2V0IC1vIGVycmV4aXQKc2V0IC1vIGVycnRyYWNlCnNldCAtbyBub3Vuc2V0CnNldCAtbyBwaXBlZmFpbAoKdXNhZ2UoKSB7CiAgICBlY2hvICIKVXNhZ2U6CiAgICAkezAjIyovfQoKT3B0aW9uYWwgYXJndW1lbnRzOgogICAgLWQsIC0tZGVidWcKICAgICAgICBBY3RpdmF0ZSB0cmFjaW5nL2RlYnVnIG1vZGUuCiAgICAtaCwgLS1oZWxwCiAgICAgICAgRGlzcGxheSB0aGlzIG1lc3NhZ2UuCgpFeGFtcGxlOgogICAgXCQgZXhwb3J0IE5BTUVTUEFDRT1cIm5hbWVzcGFjZVwiCiAgICBcJCBleHBvcnQgUkVTT1VSQ0VfVFlQRT1cImRlcGxveW1lbnRcIgogICAgJHswIyMqL30gPFJFU09VUkNFX1NFTEVDVE9SUz4KIiA+JjIKfQoKcGFyc2VfYXJncygpIHsKICAgIFJFU09VUkNFX1NFTEVDVE9SUz0oKQogICAgd2hpbGUgW1sgJCMgLWd0IDAgXV07IGRvCiAgICAgICAgY2FzZSAiJDEiIGluCiAgICAgICAgLWQgfCAtLWRlYnVnKQogICAgICAgICAgICBzZXQgLXgKICAgICAgICAgICAgREVCVUc9Ii0tZGVidWciCiAgICAgICAgICAgIGV4cG9ydCBERUJVRwogICAgICAgICAgICBpbmZvICJSdW5uaW5nIHNjcmlwdCBhczogJChpZCkiCiAgICAgICAgICAgIDs7CiAgICAgICAgLWggfCAtLWhlbHApCiAgICAgICAgICAgIHVzYWdlCiAgICAgICAgICAgIGV4aXQgMAogICAgICAgICAgICA7OwogICAgICAgICopCiAgICAgICAgICAgICMgVGhlICJyb2xsb3V0IHN0YXR1cyIgc2VsZWN0b3JzLCB0byBmaW5kIHRoZSBhY3R1YWwgcmVzb3VyY2UgdG8gY2hlY2sgZm9yCiAgICAgICAgICAgICMgc3VjY2Vzc2Z1bCByb2xsb3V0LgogICAgICAgICAgICBSRVNPVVJDRV9TRUxFQ1RPUlMrPSgiJDEiKQogICAgICAgICAgICA7OwogICAgICAgIGVzYWMKICAgICAgICBzaGlmdAogICAgZG9uZQp9CgpmYWlsKCkgewogICAgZWNobyAiIyBbRVJST1JdICR7Kn0iID4mMgogICAgZXhpdCAxCn0KCndhcm5pbmcoKSB7CiAgICBlY2hvICIjIFtXQVJOSU5HXSAkeyp9Igp9CgppbmZvKCkgewogICAgZWNobyAiIyBbSU5GT10gJHsqfSIKfQoKIwojIEZ1bmN0aW9ucwojCgpyb2xsb3V0X3N0YXR1cygpIHsKICAgIG9jIHJvbGxvdXQgc3RhdHVzICIke1JFU09VUkNFX1RZUEV9IiBcCiAgICAgICAgLS1uYW1lc3BhY2U9IiR7TkFNRVNQQUNFfSIgXAogICAgICAgIC0td2F0Y2ggXAogICAgICAgIC0tdGltZW91dD0xMHMgXAogICAgICAgIC0tc2VsZWN0b3I9IiR7MX0iCn0KCmFzc2VydF9yZXNvdXJjZV9leGlzdHMoKSB7CiAgICBsb2NhbCBzZWxlY3Rvcj0iJHsxfSIKICAgIGxvY2FsIG91dHB1dAogICAgb3V0cHV0PSQoCiAgICAgICAgb2MgZ2V0ICIke1JFU09VUkNFX1RZUEV9IiBcCiAgICAgICAgICAgIC0tbmFtZXNwYWNlPSIke05BTUVTUEFDRX0iIFwKICAgICAgICAgICAgLS1zZWxlY3Rvcj0iJHtzZWxlY3Rvcn0iIDI+JjEKICAgICkKICAgIGxvY2FsIHN0YXR1cz0kez99CiAgICBpZiBbWyAkc3RhdHVzIC1lcSAwICYmICRvdXRwdXQgIT0gIk5vIHJlc291cmNlcyBmb3VuZCIqIF1dOyB0aGVuCiAgICAgICAgaW5mbyAiUmVzb3VyY2Ugb2YgdHlwZSAnJHtSRVNPVVJDRV9UWVBFfScgd2l0aCBzZWxlY3RvciIgXAogICAgICAgICAgICAiJyR7c2VsZWN0b3J9JyBleGlzdHMgaW4gbmFtZXNwYWNlICcke05BTUVTUEFDRX0nISIKICAgICAgICByZXR1cm4gMAogICAgZmkKCiAgICB3YXJuaW5nICJSZXNvdXJjZSBvZiB0eXBlICcke1JFU09VUkNFX1RZUEV9JyB3aXRoIHNlbGVjdG9yIiBcCiAgICAgICAgIicke3NlbGVjdG9yfScgZG9lcyBub3QgZXhpc3QgaW4gbmFtZXNwYWNlICcke05BTUVTUEFDRX0nLiIKICAgIHJldHVybiAxCn0KCndhaXRfZm9yX3Jlc291cmNlKCkgewogICAgZm9yIHMgaW4gIiR7UkVTT1VSQ0VfU0VMRUNUT1JTW0BdfSI7IGRvCiAgICAgICAgZWNobyAiIyBDaGVja2luZyBpZiAke1JFU09VUkNFX1RZUEV9IHdpdGggc2VsZWN0b3IgJyR7c30nIGV4aXN0cy4uLiIKICAgICAgICBpZiAhIGFzc2VydF9yZXNvdXJjZV9leGlzdHMgIiR7c30iOyB0aGVuCiAgICAgICAgICAgIHJldHVybiAxCiAgICAgICAgZmkKCiAgICAgICAgZWNobyAiIyBDaGVja2luZyBpZiAke1JFU09VUkNFX1RZUEV9IHdpdGggc2VsZWN0b3IgJyR7c30nIGlzIHJlYWR5Li4uIgogICAgICAgIGlmICEgcm9sbG91dF9zdGF0dXMgIiR7c30iOyB0aGVuCiAgICAgICAgICAgIGVjaG8gLWVuICIjXG4jIFdBUk5JTkc6ICR7UkVTT1VSQ0VfVFlQRX0gJyR7c30nIGlzIG5vdCByZWFkeSFcbiNcbiIKICAgICAgICAgICAgcmV0dXJuIDEKICAgICAgICBmaQogICAgICAgIGluZm8gIiR7UkVTT1VSQ0VfVFlQRX0gb2JqZWN0cyB3aXRoICcke3N9JyBzZWxlY3RvciBhcmUgcmVhZHkhIgogICAgZG9uZQogICAgcmV0dXJuIDAKfQoKdGVzdF9yb2xsb3V0X3N0YXR1cygpIHsKICAgIFtbIC16ICIke05BTUVTUEFDRX0iIF1dICYmIHVzYWdlCiAgICBbWyAteiAiJHtSRVNPVVJDRV9UWVBFfSIgXV0gJiYgdXNhZ2UKICAgIFtbICR7I1JFU09VUkNFX1NFTEVDVE9SU1tAXX0gLWVxIDAgXV0gJiYgdXNhZ2UKCiAgICBmb3IgaSBpbiAkKHNlcSAxICIke1JFVFJJRVN9Iik7IGRvCiAgICAgICAgd2FpdD0kKChpICogNSkpCiAgICAgICAgW1sgJHdhaXQgLWd0IDMwICBdXSAmJiB3YWl0PTMwCiAgICAgICAgZWNobyAiIyMjIFske2l9LyR7UkVUUklFU31dIFdhaXRpbmcgZm9yICR7d2FpdH0gc2Vjb25kcyBiZWZvcmUgcmV0cnlpbmcuLi4iCiAgICAgICAgc2xlZXAgJHt3YWl0fQoKICAgICAgICBpZiB3YWl0X2Zvcl9yZXNvdXJjZTsgdGhlbgogICAgICAgICAgICBpbmZvICIke1JFU09VUkNFX1RZUEV9IG9iamVjdHMgcmVhZHk6ICcke1JFU09VUkNFX1NFTEVDVE9SU1sqXX0nIgogICAgICAgICAgICByZXR1cm4gMAogICAgICAgIGZpCiAgICBkb25lCgogICAgZmFpbCAiJyR7UkVTT1VSQ0VfVFlQRX0nIGFyZSBub3QgcmVhZHkhIgp9CgptYWluKCkgewogICAgcGFyc2VfYXJncyAiJEAiCgogICAgIyBOYW1lc3BhY2UgdG8gY2hlY2sgZm9yICJyb2xsb3V0IHN0YXR1cyIuCiAgICBkZWNsYXJlIC1yIE5BTUVTUEFDRT0iJHtOQU1FU1BBQ0U6LX0iCiAgICAjIFJlc291cmNlIHR5cGUgZm9yICJyb2xsb3V0IHN0YXR1cyIsIGFzIGluICJzdGF0ZWZ1bHNldCIgb3IgImRlcGxveW1lbnQiLgogICAgZGVjbGFyZSAtciBSRVNPVVJDRV9UWVBFPSIke1JFU09VUkNFX1RZUEU6LXN0YXRlZnVsc2V0fSIKICAgICMgTnVtYmVyIG9mIHJldHJpZXMgdG8gYXR0ZW1wdCBiZWZvcmUgZ2l2aW5nIHVwLgogICAgZGVjbGFyZSAtciBSRVRSSUVTPSR7UkVUUklFUzotMjB9CgogICAgdGVzdF9yb2xsb3V0X3N0YXR1cwp9CgppZiBbICIke0JBU0hfU09VUkNFWzBdfSIgPT0gIiQwIiBdOyB0aGVuCiAgICBtYWluICIkQCIKICAgIGVjaG8KICAgIGVjaG8gIlN1Y2Nlc3MiCmZp + base64 -d + chmod +x test-rollout-status.sh