OpenShift can be configured to use temporary credentials for workload components with Azure AD Workload Identity. In this model, the OpenShift cluster signs Kubernetes ServiceAccount tokens which can be trusted by an external OIDC identity provider. A component such as an operator running on the cluster can exchange the signed ServiceAccount token mounted into its Pod volume for an Azure Active Directory (AD) access token using Azure Identity SDKs. This process also automates requesting and refreshing of credentials. The ServiceAccount token used by a Pod is rotated by the kubelet on the node where Pod is running when the Pod's ServiceAccount token is approaching expiry. ServiceAccount tokens are valid for one hour.
This diagram illustrates how it works.
In order to take advantage of Azure AD Workload Identity, an OpenShift cluster must be configured with an RSA key pair with which to sign ServiceAccount tokens. We begin by creating this RSA key pair. The private key is provided to the OpenShift cluster via the OpenShift installer. The public key is stored in an Azure Blob Container along with OpenID configuration which serves as the identity provider.
Azure User-Assigned Managed Identities are configured with the Issuer URL (public URL of the Azure Blob Container) containing the public key and OpenID configuration to allow for federation of the identity to the OIDC identity provider. Every in-cluster component receives a Kubernetes Secret containing the details of a User-Assigned Managed Identity created on the component's behalf. The Azure Identity SDK used within those components requests an Azure AD access token for the identity specified by the configuration secret and provides the signed ServiceAccount token to Azure AD. Azure AD validates that the ServiceAccount token is from a trusted identity provider before issuing an access token for the User-Assigned Managed Identity.
Without Azure AD Workload Identity, an Azure credentials secret includes an azure_client_secret
which is essentially a password used to authenticate with Azure AD for the identity identified by azure_client_id
. This azure_client_secret
needs to be kept secure and is not rotated automatically.
apiVersion: v1
data:
azure_client_id: <client id>
azure_client_secret: <client secret>
azure_region: <region>
azure_resource_prefix: <resource group prefix eg. "mycluster-az-t68n4">
azure_resourcegroup: <resource group eg. "mycluster-az-t68n4-rg">
azure_subscription_id: <subscription id>
azure_tenant_id: <tenant id>
kind: Secret
type: Opaque
With Azure AD Workload Identity, an Azure credentials secret contains the azure_client_id
of the User-Assigned Managed Identity that the component will be authenticating as along with the path to the mounted ServiceAccount token, azure_federated_token_file
. The ServiceAccount token mounted at this path is refreshed hourly and thus the credentials are short-lived.
apiVersion: v1
data:
azure_client_id: <client id>
azure_federated_token_file: </path/to/mounted/service/account/token>
azure_region: <region>
azure_subscription_id: <subscription id>
azure_tenant_id: <tenant id>
kind: Secret
type: Opaque
-
Obtain a recent version of the OpenShift CLI
oc
.Reference the OpenShift CLI installation steps in the OpenShift documentation.
-
Set the
$RELEASE_IMAGE
environment variable.$RELEASE_IMAGE
should be a recent and supported OpenShift release image that you want to deploy in your cluster. Please refer to the support matrix for compatibilities.A sample release image would be
RELEASE_IMAGE=quay.io/openshift-release-dev/ocp-release:${RHOCP_version}-${Arch}
Where
RHOCP_version
is the OpenShift version (e.g4.14.0-fc.4
or4.14.3
) and theArch
is the architecture type (e.gx86_64
). -
Extract Azure CredentialsRequests objects from the release image.
oc adm release extract --cloud=azure --credentials-requests $RELEASE_IMAGE --to=./credreqs
-
Extract the
openshift-install
andccoctl
binaries from the release image.oc adm release extract --command=openshift-install $RELEASE_IMAGE --registry-config ~/.pull-secret oc adm release extract --command=ccoctl $RELEASE_IMAGE --registry-config ~/.pull-secret
-
Create Azure resources using the ccoctl tool. You will need Azure credentials with sufficient permissions. The Azure credentials can be automatically detected after having logged into the Azure CLI
az login
or may be provided as environment variables (AZURE_CLIENT_ID
,AZURE_CLIENT_SECRET
,AZURE_TENANT_ID
).The following command will,
- Generate public/private ServiceAccount signing RSA keys.
- Create an empty Azure ResourceGroup in which to install the cluster. This ResourceGroup is used to scope created identities and must be configured as the cluster installation ResourceGroup within the
install-config.yaml
. - Create an Azure ResourceGroup in which to create identity resources.
- Create an Azure StorageAccount, Azure Blob Container and upload OIDC configuration to the Blob Container.
- Create User-Assigned Managed Identities for each Azure CredentialsRequest.
Manifest files needed for cluster installation are output to the
ccoctl_output_dir
directory../ccoctl azure create-all --name <azure_infra_name> \ --output-dir <ccoctl_output_dir> \ --region <azure_region> \ --subscription-id <azure_subscription_id> \ --tenant-id <azure_tenant_id> \ --credentials-requests-dir /path/to/credreqs/directory/created/in/step/3 \ --dnszone-resource-group-name <azure_dns_zone_resourcegroup_name>
-
Create an install-config.yaml with the extracted OpenShift installer binary. Be sure to choose the same region as specified in step 5.
./openshift-install create install-config
-
Make sure that we install the cluster in Manual mode by setting the
credentialsMode
withininstall-config.yaml
.echo "credentialsMode: Manual" >> install-config.yaml
-
Configure the installation Azure ResourceGroup created in step 5 within the
install-config.yaml
. By default, the installation Azure ResourceGroup will be the<azure_infra_name>
used in step 5.The
install-config.yaml
must be modified to add theresourceGroupName
key within Azure platform (.platform.azure.resourceGroupName
). We can accomplish this using yq or by editinginstall-config.yaml
directly.yq -i '.platform.azure.resourceGroupName = "<azure_infra_name>"' install-config.yaml
-
Create install manifests.
./openshift-install create manifests
-
Copy the credentials manifests created in step 5 and place them in the install
manifests
directory generated byopenshift-install create manifests
.
cp ccoctl_output_dir/manifests/* /path/to/dir/with/install/manifests/
- Copy the private key for the ServiceAccount signer and place it in the same location as
install-config.yaml
.
cp -a ccoctl_output_dir/tls /path/to/dir/with/install-config.yaml
- Run the OpenShift installer.
./openshift-install create cluster --log-level=debug
-
Connect to the newly installed cluster and verify that the OpenShift cluster does not have
root
credentials.This command should return a secret not found error.
oc get secrets -n kube-system azure-credentials
-
Verify that components are assuming the
azure_client_id
specified in the secret manifests, instead of credentials passed through by the Cloud Credential Operator. The secret displayed should not contain anazure_client_secret
key and will instead contain anazure_federated_token_file
key.oc get secrets -n openshift-image-registry installer-cloud-credentials -o yaml
Sample output of the above command:
{ "azure_client_id": "cG90YXRvIHNhbGFkIGJyZWFkIGJvd2wK", "azure_federated_token_file": "L2RyeS9ydWIvY2hpY2tlbi93aW5ncy8K", "azure_region": "Y29ybiBicmVhZAo=", "azure_subscription_id": "Y2hvcml6byB0YWNvcwo=", "azure_tenant_id": "bW9sYXNzZXMgY29va2llcwo=" }
Make sure you clean up the following resources after you uninstall your cluster. You can use the <azure_infra_name>
used in installation step 5 to identify these resources.
./ccoctl azure delete --name <azure_infra_name> \
--region <azure_region> \
--subscription-id <azure_subscription_id> \
--delete-oidc-resource-group
NOTE This is just for developers interested in taking an existing cluster to Azure AD Workload Identity. This is explicitly NOT RECOMMENED OR SUPPORTED.
-
Extract the cluster's ServiceAccount public signing key.
oc get configmap --namespace openshift-kube-apiserver bound-sa-token-signing-certs --output json | jq --raw-output '.data["service-account-001.pub"]' > serviceaccount-signer.public
-
Create an output directory for
ccoctl
generated manifests and move the public key file to the output directory.mkdir ./output_dir ; mv serviceaccount-signer.public ./output_dir/serviceaccount-signer.public
-
Create an OIDC issuer and Azure blob storage container with OIDC configuration files, providing the cluster's existing ServiceAccount public signing key which we extracted.
Note that an Azure resource group will be created with a name matching the provided
<azure_infra_name>
. An existing Azure resource group can be optionally specified by providing the--oidc-resource-group-name
parameter.Ensure that the region and subscription match that of the existing cluster.
./ccoctl azure create-oidc-issuer --name <azure_infra_name> \ --output-dir output_dir \ --region <azure_region> \ --subscription-id <azure_subscription_id> \ --tenant-id <azure_tenant_id> \ --public-key-file output_dir/serviceaccount-signer.public
-
Extract the OIDC issuer URL from the generated manifests in the output directory and patch the cluster
authentication
config, settingspec.serviceAccountIssuer
.OIDC_ISSUER_URL=`awk '/serviceAccountIssuer/ { print $2 }' output_dir/manifests/cluster-authentication-02-config.yaml` oc patch authentication cluster --type=merge -p "{\"spec\":{\"serviceAccountIssuer\":\"${OIDC_ISSUER_URL}\"}}"
-
Wait for the kube-apiserver pods to be updated with the new config. This process can take several minutes.
oc get pods -n openshift-kube-apiserver | grep kube-apiserver
-
Restart all pods (this will take a while) in the cluster (because all ServiceAccounts need to be refreshed after updating the serviceAccountIssuer field):
for I in $(oc get ns -o jsonpath='{range .items[*]} {.metadata.name}{"\n"} {end}'); \ do oc delete pods --all -n $I; \ sleep 1; \ done
-
Set the `CloudCredentials`` CR's .spec.credentialsMode to Manual.
oc patch cloudcredential cluster --patch '{"spec":{"credentialsMode":"Manual"}}' --type=merge
-
Extract CredentialsCrequests from the cluster's release image for the given version.
# Obtain release image from the cluster version. Will not work with pre-release versions. CLUSTER_VERSION=`oc get clusterversion version -o json | jq -r '.status.desired.version'` RELEASE_IMAGE=`oc get clusterversion version -o json | jq -r '.status.history[] | select(.version == "VERSION_FROM_PREVIOUS_COMMAND") | .image'` # Set the RELEASE_IMAGE explicitly. RELEASE_IMAGE="registry.ci.openshift.org/ocp/release:4.14.0-0.ci-2023-09-12-113607" oc adm release extract --credentials-requests --cloud=azure $RELEASE_IMAGE --to credreqs
-
Create Managed Identities for each of the CredentialsRequests from the release image.
AZURE_INSTALL_RG=`oc get infrastructure cluster -o jsonpath --template '{ .status.platformStatus.azure.resourceGroupName }'` ./ccoctl azure create-managed-identities --name <azure_infra_name> \ --output-dir output_dir \ --region <azure_region> \ --subscription-id <azure_subscription_id> \ --credentials-requests-dir ./credreqs \ --issuer-url "${OIDC_ISSUER_URL}" \ --dnszone-resource-group-name <azure_dns_zone_resourcegroup_name> \ --installation-resource-group-name "${AZURE_INSTALL_RG}"
-
Apply the azure pod identity webhook configuration.
oc replace -f output_dir/manifests/azure-ad-pod-identity-webhook-config.yaml
- Apply Secrets generated by the creating managed identities.
find ./output_dir/manifests -iname "openshift*yaml" -print0 | xargs -I {} -0 -t oc replace -f {}
- At this point the cluster is using Azure AD Workload Identity. The "root" Azure credentials Secret can be removed.
oc delete secret -n kube-system azure-credentials