diff --git a/common/oidc-client/oauth2-proxy/components/configure-self-signed-kubernetes-oidc-issuer/cronjob.kubeflow-m2m-oidc-configurator.yaml b/common/oidc-client/oauth2-proxy/components/configure-self-signed-kubernetes-oidc-issuer/cronjob.kubeflow-m2m-oidc-configurator.yaml index c98304bf03..951b8dac4f 100644 --- a/common/oidc-client/oauth2-proxy/components/configure-self-signed-kubernetes-oidc-issuer/cronjob.kubeflow-m2m-oidc-configurator.yaml +++ b/common/oidc-client/oauth2-proxy/components/configure-self-signed-kubernetes-oidc-issuer/cronjob.kubeflow-m2m-oidc-configurator.yaml @@ -14,6 +14,7 @@ spec: labels: {} spec: restartPolicy: OnFailure + backoffLimit: 3 serviceAccountName: kubeflow-m2m-oidc-configurator containers: - image: docker.io/curlimages/curl diff --git a/common/oidc-client/oauth2-proxy/components/configure-self-signed-kubernetes-oidc-issuer/script.sh b/common/oidc-client/oauth2-proxy/components/configure-self-signed-kubernetes-oidc-issuer/script.sh index a3348d95e0..bef022edda 100644 --- a/common/oidc-client/oauth2-proxy/components/configure-self-signed-kubernetes-oidc-issuer/script.sh +++ b/common/oidc-client/oauth2-proxy/components/configure-self-signed-kubernetes-oidc-issuer/script.sh @@ -12,72 +12,132 @@ ${ISTIO_ROOT_NAMESPACE}\ /requestauthentications/\ ${REQUEST_AUTHENTICATION_NAME}" -echo "Wait until resource RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} in namespace ${ISTIO_ROOT_NAMESPACE} is ready." -while true; do - response="$( - curl -s -o /dev/null \ - --url "${RESOURCE_URL}" \ - -w "%{http_code}" \ - --header "Authorization: Bearer $(cat /run/secrets/kubernetes.io/serviceaccount/token)" \ - --insecure - )" - if [ "${response}" = "200" ]; then - break - fi - echo "Resource RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} in namespace ${ISTIO_ROOT_NAMESPACE} is not ready yet." - sleep 5 -done -echo "Resource RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} in namespace ${ISTIO_ROOT_NAMESPACE} is ready." +wait_for_resource_ready() { + while true; do + response="$( + curl -s -o /dev/null \ + --url "${RESOURCE_URL}" \ + -w "%{http_code}" \ + --header "Authorization: Bearer $(cat /run/secrets/kubernetes.io/serviceaccount/token)" \ + --insecure + )" + if [ "${response}" = "200" ]; then + break + fi + sleep 5 + done +} -echo "Getting RequestAuthentication object." -REQUEST_AUTHENTICATION_OBJ="$( +get_request_authentication_obj() { curl -s --request GET \ --url "${RESOURCE_URL}" \ --header "Authorization: Bearer $(cat /run/secrets/kubernetes.io/serviceaccount/token)" \ --insecure -)" +} -# Get Issuer URL configured in RequestAuthentication. -ISSUER_URL="$(echo "${REQUEST_AUTHENTICATION_OBJ}" | awk -F'"' '/"issuer":/ { print $4 }')" -echo "Issuer Url in RequestAuthentication: ${ISSUER_URL}" +get_issuer_url_from_obj() { + obj="${1}" + echo "${obj}" | awk -F'"' '/"issuer":/ { print $4 }' +} -CURRENT_JWKS_ESCAPED="$(echo "${REQUEST_AUTHENTICATION_OBJ}" | awk -F'"' '/"jwks":/' | sed -n 's/^.*"jwks": "\(.*\)".*$/\1/p')" -printf "Current Jwks (escaped):\n%s\n" "${CURRENT_JWKS_ESCAPED}" +get_current_escaped_jwks_from_obj() { + obj="${1}" + echo "${obj}" | awk -F'"' '/"jwks":/' | sed -n 's/^.*"jwks": "\(.*\)".*$/\1/p' +} -# GET URI to the JWKS. -JWKS_URI="$( +get_jwks_uri() { + issuer_url="${1}" curl -s --request GET \ - --url "${ISSUER_URL}/.well-known/openid-configuration" \ - --header "Authorization: Bearer $(cat /run/secrets/kubernetes.io/serviceaccount/token)" \ - --insecure | - grep -o '"jwks_uri":"https:\/\/[^"]\+"' | - sed 's/"jwks_uri":"\(.*\)"/\1/' -)" -echo "Jwks Uri from Well Known OpenID Configuration: ${JWKS_URI}" - -# Get content of the JWKS. -JWKS="$( + --url "${issuer_url}/.well-known/openid-configuration" \ + --header "Authorization: Bearer $(cat /run/secrets/kubernetes.io/serviceaccount/token)" \ + --insecure | + grep -o '"jwks_uri":"https:\/\/[^"]\+"' | + sed 's/"jwks_uri":"\(.*\)"/\1/' + } + +get_jwks_from_uri() { + jwks_uri="${1}" curl -s --request GET \ - --url "${JWKS_URI}" \ + --url "${jwks_uri}" \ + --header "Authorization: Bearer $(cat /run/secrets/kubernetes.io/serviceaccount/token)" \ + --insecure +} + +# Format JWKS in a way that can be accepted in resource patch. +parse_escaped_jwks() { + jwks="${1}" + echo "${jwks}" | sed 's/"/\\"/g' +} + +are_jwks_equal() { + jwks1="${1}" + jwks2="${2}" + test "$(echo "${jwks1}" | base64 -w0)" = "$(echo "${jwks2}" | base64 -w0)" +} + +patch_request_authentication_with_escaped_jwks() { + jwks_escaped="${1}" + curl -s --request PATCH \ + --url "${RESOURCE_URL}" \ + --header "Content-Type: application/json-patch+json" \ --header "Authorization: Bearer $(cat /run/secrets/kubernetes.io/serviceaccount/token)" \ + -d '[{ "op": "add", "path": "/spec/jwtRules/0/jwks", "value": "'"${jwks_escaped}"'" }]' \ --insecure -)" + echo +} -# Format JWKS in a way that can be accepted in resource patch. -JWKS_ESCAPED="$(echo "${JWKS}" | sed 's/"/\\"/g')" -printf "JWKS from Well Known OpenID Configuration (escaped): \n%s\n" "${JWKS_ESCAPED}" - -# If the JWKS from RequestAuthentication and OpenID Configuration is the same, don't to any changes. -if [ "$(echo "${JWKS_ESCAPED}" | base64 -w0)" = "$(echo "${CURRENT_JWKS_ESCAPED}" | base64 -w0)" ]; then - echo "JWKS in RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} is configured correctly, exitting..." - exit 0 -fi -echo "JWKS in RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} needs to be configured." - -# Patch the RequestAuthentication with JWKS. -curl -s --request PATCH \ - --url "${RESOURCE_URL}" \ - --header "Content-Type: application/json-patch+json" \ - --header "Authorization: Bearer $(cat /run/secrets/kubernetes.io/serviceaccount/token)" \ - -d '[{ "op": "add", "path": "/spec/jwtRules/0/jwks", "value": "'"${JWKS_ESCAPED}"'" }]' \ - --insecure +patch_request_authentication_with_jwks_if_required() { + echo "Getting RequestAuthentication object." + REQUEST_AUTHENTICATION_OBJ="$(get_request_authentication_obj)" + + ISSUER_URL="$(get_issuer_url_from_obj "${REQUEST_AUTHENTICATION_OBJ}")" + echo "Issuer Url in RequestAuthentication: ${ISSUER_URL}" + + CURRENT_JWKS_ESCAPED="$(get_current_escaped_jwks_from_obj "${REQUEST_AUTHENTICATION_OBJ}")" + printf "Current Jwks (escaped):\n%s\n" "${CURRENT_JWKS_ESCAPED}" + + JWKS_URI="$(get_jwks_uri "${ISSUER_URL}")" + echo "Jwks Uri from Well Known OpenID Configuration: ${JWKS_URI}" + + JWKS="$(get_jwks_from_uri "${JWKS_URI}")" + JWKS_ESCAPED="$(parse_escaped_jwks "${JWKS}")" + printf "JWKS from Well Known OpenID Configuration (escaped): \n%s\n" "${JWKS_ESCAPED}" + + if are_jwks_equal "${JWKS_ESCAPED}" "${CURRENT_JWKS_ESCAPED}"; then + echo "JWKS in RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} is configured correctly." + else + echo "JWKS in RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} needs to be configured." + patch_request_authentication_with_escaped_jwks "${JWKS_ESCAPED}" + fi +} + +verify_jwks_in_request_authentication() { + REQUEST_AUTHENTICATION_OBJ="$(get_request_authentication_obj)" + ISSUER_URL="$(get_issuer_url_from_obj "${REQUEST_AUTHENTICATION_OBJ}")" + CURRENT_JWKS_ESCAPED="$(get_current_escaped_jwks_from_obj "${REQUEST_AUTHENTICATION_OBJ}")" + JWKS_URI="$(get_jwks_uri "${ISSUER_URL}")" + JWKS="$(get_jwks_from_uri "${JWKS_URI}")" + JWKS_ESCAPED="$(parse_escaped_jwks "${JWKS}")" + if ! are_jwks_equal "${JWKS_ESCAPED}" "${CURRENT_JWKS_ESCAPED}"; then + echo "JWKS not properly configured, exit with error code 1" + exit 1 + fi +} + +main() { + echo "Wait until resource RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} in namespace ${ISTIO_ROOT_NAMESPACE} is ready." + wait_for_resource_ready + echo "Resource RequestAuthentication ${REQUEST_AUTHENTICATION_NAME} in namespace ${ISTIO_ROOT_NAMESPACE} is ready." + + echo "Patch RequestAuthentication with JWKS if required." + patch_request_authentication_with_jwks_if_required + + echo "Wait 5 seconds before verifying RequestAuthentication JWKS configuration." + sleep 5 + + echo "Verify if RequestAuthentication is properly configured with JWKS..." + verify_jwks_in_request_authentication + echo "RequestAuthentication is properly configured with JWKS." +} + +main