- Introduction
- Configure a Keycloak instance for ODM (Part 1)
- Deploy ODM on a container configured with Keycloak (Part 2)
- Troubleshooting
- License
In the context of the Operational Decision Manager (ODM) on Certified Kubernetes offering, ODM for production can be configured with an external OpenID Connect server (OIDC provider), such as the Keycloak cloud service.
This tutorial shows how to integrate ODM with Keycloak to manage classic authentication and authorization. Another tutorial explains how to manage fine grain permission with Decision Center by using a Keycloak SCIM Server.
Keycloak is an open source enterprise identity service that provides single sign-on, user federation, identity brokering and social login. The Keycloak SSO OpenID Connect capability is the service that we use in this article.
You need to create a number of secrets before you can install an ODM instance with an external OIDC provider such as the Keycloak service, and use web application single sign-on (SSO). The following diagram shows the ODM services with an external OIDC provider after a successful installation.
The following procedure describes how to manually configure ODM with a Keycloak service.
OpenID Connect is an authentication standard built on top of OAuth 2.0. It adds a token called an ID token.
Terminology:
- The OpenID provider — The authorization server that issues the ID token. In this case, Keycloak is the OpenID provider.
- The end user — The end user whose information is contained in the ID token.
- The relying party — The client application that requests the ID token from Keycloak.
- The ID token — The token that is issued by the OpenID provider and contains information about the end user in the form of claims.
- A claim — A piece of information about the end user.
The Authorization Code flow is best used by server-side apps where the source code is not publicly exposed. The apps must be server-side because the request that exchanges the authorization code for a token requires a client secret, which has to be stored in your client. However, the server-side app requires an end user because it relies on interactions with the end user's web browser, which redirects the user and then receives the authorization code.
The Client Credentials flow is intended for server-side (AKA "confidential") client applications with no end user, which normally describes machine-to-machine communication. The application must be server-side because it must be trusted with the client secret, and since the credentials are hard-coded, it cannot be used by an actual end user. It involves a single, authenticated request to the token endpoint, which returns an access token.
The resource owner password flow allows an application to sign in a user by directly handling their password. It is not recommended to use this flow. In most scenarios, more secure alternatives are available and recommended. This flow requires a very high degree of trust in the application, and carries risks which are not present in other flows. You should only use this flow when other more secure flows cannot be used.
You need the following elements:
- Helm v3
- Kubectl
- Access to an Operational Decision Manager product
- Access to a CNCF Kubernetes cluster
- A Keycloak Instance
This tutorial has been tested with a Keycloak instance version 26.0.5 installed on Openshift. The installation procedure can be found at Get started with Keycloak on Openshift. If you already have an Openshift cluster, you can skip the section Before you start and use the following steps:
- oc login to your cluster
- Create a namespace "keycloak":
oc new-project keycloak
- Continue from the section Start Keycloak using the file keycloak.yaml instead of the one online as suggested at the first step (which may prevent from accessing the console behind a proxy), ie.:
oc process -f keycloak.yaml \ -p KEYCLOAK_ADMIN=admin \ -p KEYCLOAK_ADMIN_PASSWORD=admin \ -p NAMESPACE=keycloak \ | oc create -f -
If you want to install on another Kubernetes platform, follow these instructions: Get started with Keycloak on Kubernetes.
In this section, we explain how to:
- Log into the Keycloak Admin Console
- Create a dedicated odm realm
- Manage roles, groups, and users
- Set up an application
- Configure the default Authorization server
Using OpenShift routes, you should be able to access the Keycloak Admin Console through the following URL with the admin username and admin password:
KEYCLOAK_URL=https://$(oc get route keycloak --template='{{ .spec.host }}') &&
echo "" &&
echo "Keycloak Admin Console: $KEYCLOAK_URL/admin" &&
echo ""
The following configuration takes place in this Admin Console.
This step is not compulsory as you can perform the following tasks in the default master realm. But to avoid getting mixing up with existing configurations, it is preferable to create a dedicated odm realm.
-
Create an odm realm
On the Main page, click Master:
- Click Create Realm
- Realm Name: odm
- Enabled: On
- Click Create
- Click Create Realm
As explained in Keycloak documentation, groups and roles enables to give users access and permissions to use applications. Groups are collections of users to which you apply roles and attributes. Roles define specific application permissions and access control. To manage permissions inside the ODM application, the ID token and access token contain a property (aka claim) named 'groups' listing all the roles granted to the identified user. You can create roles and grant these roles directly to an individual user, or even better to a group. This way, all the users of the group are granted the roles of the group.
-
Create a role for ODM administrators.
In Menu Manage / Realm roles:
- Click Create role
- Role name: rtsAdministrators
- Click Save
Do the same for the other ODM J2EE roles:
- rtsConfigManagers
- rtsInstallers
- rtsUsers
- resAdministrators
- resMonitors
- resDeployers
- resExecutors
For more information about ODM groups and roles, refer to the ODM on Kubernetes documentation.
- Click Create role
-
Create a group for ODM administrators.
In Menu Manage / Groups:
- Click Create group
- Name: odm-admin
In Menu Manage / Groups:
- Click Create odm-admin
- Click the Role mapping tab
- Click Assign role
- Select "Filter by realm roles"
- Select all previously created ODM roles
- Click Assign
- Click Assign role
- Click Create group
-
Create at least one user that belongs to this new group.
In Menu Manage / Users:
- Click Create new user
- Username:
johndoe@mynicecompany.com
- Email:
johndoe@mynicecompany.com
- Email Verified: On
- First name:
John
- Last name:
Doe
- Enabled: On
- Required user actions: nothing
- Groups : Click Join Groups , select odm-admin, and click Join
- Click Create
- Username:
- In User Details, select the Credentials tab
- Click Set password
- Fill the Password and Password confirmation fields with johndoe
- Temporary: Off
- Click Save Password
- Click the Details tab
- Click Save
(Optional) Every user is created with a predefined role named default-roles-<CLIENT_ID>. This role has no interest. So, here is the way to unassign this role.
- In User Details, select the Role mapping tab
- Select default-roles-<CLIENT_ID>
- Click Unassign
- Click Remove
- Click the Details tab
- Click Save
Repeat those steps for each user you want to add.
- Click Create new user
-
Create the ODM client.
In Menu Manage / Clients, click Create client:
- Client type: OpenID Connect
- Client ID: odm
- Name: ODM Application
- Always display in console: On
- Click Next
- Client Authentication: On
- Authorization: On
- Click Next or Save
- Newer versions of Keycloak display the additional Login settings page below. If you can see it, just click Save.
- Click the Credentials tab
- Take a note of the Client secret value. It will be referenced as
CLIENT_SECRET
in the next steps.
- Click the Service account roles tab
- Click the Assign role button.
- Select Filter by realm roles
- Select all res* and rts* roles in the list and click the Assign button.
-
Add the GROUPS predefined mapper on the ROLES client scope
- Select the Manage / Client scopes menu
- click the roles scope
- Select the Mappers tab
- Click Add mapper>From predefined mappers
- Search for mapper : groups
- Select groups
- Click Add
- Select the Manage / Client scopes menu
-
Retrieve the Keycloak Server URL
In Menu Configure/Realm settings, in the General tab, click the OpenID Endpoint Configuration link. Take note of the issuer URL. It will be referenced as
KEYCLOAK_SERVER_URL
in the next steps. -
Check the configuration
Download the keycloak-odm-script.zip file to your machine and unzip it in your working directory. This .zip file contains scripts and templates to verify and set up ODM.
4.1 Verify the Client Credentials Token
You can request an access token using the Client-Credentials flow to verify the format of the token. This token is used for the deployment between Decision Center and the Decision Server Console:
./get-client-credential-token.sh -i $CLIENT_ID -x $CLIENT_SECRET -n $KEYCLOAK_SERVER_URL
Where:
- CLIENT_ID is your ODM Application, default is odm, can be retrieved in the Manage / Clients menu
- CLIENT_SECRET is listed in your ODM Application, in the Credentials tab
- KEYCLOAK_SERVER_URL is the issuer that can be retrieved using the OpenID Endpoint Configuration link of the General tab in the Configure/Realm settings menu
By introspecting the access_token value with the online tool https://jwt.io, you should get:
{ .. "iss": "<KEYCLOAK_SERVER_URL>", .... "preferred_username": "service-account-<CLIENT_ID>", ... }
4.2 Verify the Client Password Token
To check that it has been correctly taken into account, you can request an access token using the Client password flow. This token is used for the invocation of the ODM components like the Decision Center, Decision Server console, and the invocation of the Decision Server Runtime REST API.
./get-user-password-token.sh -i $CLIENT_ID -x $CLIENT_SECRET -n $KEYCLOAK_SERVER_URL -u <USERNAME> -p <PASSWORD>
Where:
- CLIENT_ID is your ODM Application, default is odm, can be retrieved in the Manage / Clients menu
- CLIENT_SECRET is listed in your ODM Application, in the Credentials tab
- KEYCLOAK_SERVER_URL is the issuer that can be retrieved using the OpenID Endpoint Configuration link of the General tab in the Configure/Realm settings menu
- USERNAME and PASSWORD have been created from 'Create at least one user that belongs to this new group.' section.
By introspecting the id_token value with the online tool https://jwt.io, you should get:
{ .. "iss": "<KEYCLOAK_SERVER_URL>", .... "preferred_username": "<USERNAME>", "groups": [ "rtsAdministrators", "rtsInstallers", "rtsConfigManagers", "rtsUsers", "resAdministrators", "resDeployers", "resMonitors", "resExecutors" ], ... }
-
To get your entitlement key, log in to MyIBM Container Software Library with the IBMid and password that are associated with the entitled software.
In the Container software library tile, verify your entitlement on the View library page, and then go to Get entitlement key to retrieve the key.
-
Create a pull secret by running a
kubectl create secret
command.kubectl create secret docker-registry icregistry-secret \ --docker-server=cp.icr.io \ --docker-username=cp \ --docker-password="<API_KEY_GENERATED>" \ --docker-email=<USER_EMAIL>
Where:
- API_KEY_GENERATED is the entitlement key from the previous step. Make sure you enclose the key in double-quotes.
- USER_EMAIL is the email address associated with your IBMid.
Note: The cp.icr.io value for the docker-server parameter is the only registry domain name that contains the images. You must set the docker-username to cp to use an entitlement key as docker-password.
-
Make a note of the secret name so that you can set it for the image.pullSecrets parameter when you run a helm install of your containers. The image.repository parameter is later set to cp.icr.io/cp/cp4a/odm.
-
Create a secret with the Keycloak Server certificate.
To allow ODM services to access the Keycloak Server, it is mandatory to provide the Keycloak Server certificate. You can create the secret as follows:
keytool -printcert -sslserver <KEYCLOAK_SERVER_URL_WITHOUT_HTTPS> -rfc > keycloak.crt kubectl create secret generic keycloak-secret --from-file=tls.crt=keycloak.crt
Where:
- KEYCLOAK_SERVER_URL_WITHOUT_HTTPS is KEYCLOAK_SERVER_URL without the leading
https://
- KEYCLOAK_SERVER_URL_WITHOUT_HTTPS is KEYCLOAK_SERVER_URL without the leading
-
Generate the ODM configuration file for Keycloak.
If you have not done it yet, download the keycloak-odm-script.zip file to your machine. This .zip file contains the script and the content of the templates directory. The script allows you to generate the necessary configuration files.
Generate the files with the following command:
./generateTemplate.sh -i $CLIENT_ID -x $CLIENT_SECRET -n $KEYCLOAK_SERVER_URL [-r $REALM_NAME -u $USERID_CLAIM]
Where:
- CLIENT_SECRET is listed in your ODM Application, section General / Client Credentials
- REALM_NAME is optional (odm by default)
- USERID_CLAIM is optional (preferred_username by default). This is the name of the claim (ie. parameter) in the token that holds the name of the user.
The following files are generated into the
output
directory:webSecurity.xml
contains the mapping between Liberty J2EE ODM roles and Keycloak groups and users:- rtsAdministrators/resAdministrators/resExecutors ODM roles are given to the CLIENT_ID (which is seen as a user) to manage the client-credentials flow
openIdWebSecurity.xml
contains two openIdConnectClient Liberty configurations:- for web access to Decision Center and Decision Server consoles using userIdentifier="preferred_username" with the Authorization Code flow
- for the rest-api call using userIdentifier="preferred_username" with the client-credentials flow
openIdParameters.properties
configures several features like allowed domains, logout, and some internal ODM openid featuresldap-configurations.xml
contains LDAP configuration for How to import Keycloak Groups and Users using SCIM
-
Create the Keycloak authentication secret using
webSecurity.xml
,openIdWebSecurity.xml
andopenIdParameters.properties
files.kubectl create secret generic keycloak-auth-secret \ --from-file=openIdParameters.properties=./output/openIdParameters.properties \ --from-file=openIdWebSecurity.xml=./output/openIdWebSecurity.xml \ --from-file=webSecurity.xml=./output/webSecurity.xml
helm repo add ibm-helm https://raw.githubusercontent.com/IBM/charts/master/repo/ibm-helm
helm repo update
helm search repo ibm-odm-prod
The output should look like:
NAME CHART VERSION APP VERSION DESCRIPTION
ibm-helm/ibm-odm-prod 24.1.0 9.0.0.1 IBM Operational Decision Manager
You can now install the product. We will use the PostgreSQL internal database and disable data persistence (internalDatabase.persistence.enabled=false
) to avoid any platform complexity with persistent volume allocation.
See the Preparing to install documentation for more information.
helm install my-odm-release ibm-helm/ibm-odm-prod \
--set image.repository=cp.icr.io/cp/cp4a/odm --set image.pullSecrets=icregistry-secret \
--set oidc.enabled=true \
--set license=true \
--set internalDatabase.persistence.enabled=false \
--set internalDatabase.populateSampleData=true \
--set decisionCenter.disableAllAuthenticatedUser=true \
--set customization.trustedCertificateList={"keycloak-secret"} \
--set customization.authSecretRef=keycloak-auth-secret \
--set internalDatabase.runAsUser='' --set customization.runAsUser='' --set service.enableRoute=true
Refer to the following documentation to install an NGINX Ingress Controller on:
When the NGINX Ingress Controller is ready, you can install the ODM release with:
helm install my-odm-release ibm-helm/ibm-odm-prod \
--set image.repository=cp.icr.io/cp/cp4a/odm --set image.pullSecrets=icregistry-secret \
--set oidc.enabled=true \
--set license=true \
--set internalDatabase.persistence.enabled=false \
--set internalDatabase.populateSampleData=true \
--set customization.trustedCertificateList={"keycloak-secret"} \
--set customization.authSecretRef=keycloak-auth-secret \
--set service.ingress.enabled=true \
--set decisionCenter.disableAllAuthenticatedUser=true \
--set service.ingress.annotations={"kubernetes.io/ingress.class: nginx"\,"nginx.ingress.kubernetes.io/backend-protocol: HTTPS"\,"nginx.ingress.kubernetes.io/affinity: cookie"}
-
Get the ODM endpoints. Refer to this documentation to retrieve the endpoints. For example, on OpenShift you can get the route names and hosts with:
kubectl get routes --no-headers --output custom-columns=":metadata.name,:spec.host"
You get the following hosts:
my-odm-release-odm-dc-route <DC_HOST> my-odm-release-odm-dr-route <DR_HOST> my-odm-release-odm-ds-console-route <DS_CONSOLE_HOST> my-odm-release-odm-ds-runtime-route <DS_RUNTIME_HOST>
Using an Ingress, the endpoint is the address of the ODM ingress and is the same for all components. You can get it with:
kubectl get ingress my-odm-release-odm-ingress
You get the following ingress address:
NAME CLASS HOSTS ADDRESS PORTS AGE my-odm-release-odm-ingress <none> * <INGRESS_ADDRESS> 80 14d
-
Register the redirect URIs into your Keycloak application.
The redirect URIs are built in the following way:
Using Routes:
- Decision Center redirect URI:
https://<DC_HOST>/decisioncenter/openid/redirect/odm
- Decision Runner redirect URI:
https://<DR_HOST>/DecisionRunner/openid/redirect/odm
- Decision Server Console redirect URI:
https://<DS_CONSOLE_HOST>/res/openid/redirect/odm
- Decision Server Runtime redirect URI:
https://<DS_RUNTIME_HOST>/DecisionService/openid/redirect/odm
- Rule Designer redirect URI:
https://127.0.0.1:9081/oidcCallback
Using Ingress:
- Decision Center redirect URI:
https://<INGRESS_ADDRESS>/decisioncenter/openid/redirect/odm
- Decision Runner redirect URI:
https://<INGRESS_ADDRESS>/DecisionRunner/openid/redirect/odm
- Decision Server Console redirect URI:
https://<INGRESS_ADDRESS>/res/openid/redirect/odm
- Decision Server Runtime redirect URI:
https://<INGRESS_ADDRESS>/DecisionService/openid/redirect/odm
- Rule Designer redirect URI:
https://127.0.0.1:9081/oidcCallback
From the Keycloak admin console, in Manage / Clients / odm
-
In the tab Settings
- Add the redirect URIs in the Valid redirect URIs field for each components.
For example, add the Decision Center redirect URI that you got earlier (
https://<DC_HOST>/decisioncenter/openid/redirect/odm
-- do not forget to replace <DC_HOST> with your actual host name!) -
Click Save at the bottom of the page.
- Decision Center redirect URI:
Well done! You can now connect to ODM using the endpoints you got earlier and log in as an ODM admin with the account you created in the first step (e.g. johndoe@mycompany.com/johndoe).
To be able to securely connect your Rule Designer to the Decision Server and Decision Center services that are running in Certified Kubernetes, you need to establish a TLS connection through a security certificate in addition to the OpenID configuration.
-
Get the following configuration files.
https://<DC_HOST>/decisioncenter/assets/truststore.jks
https://<DC_HOST>/decisioncenter/assets/OdmOidcProvidersRD.json
where DC_HOST is the Decision Center endpoint.
-
Copy the
truststore.jks
andOdmOidcProvidersRD.json
files to your Rule Designer installation directory next to theeclipse.ini
file. -
Edit your
eclipse.ini
file and add the following lines at the end.-Djavax.net.ssl.trustStore=<ECLIPSEINITDIR>/truststore.jks -Djavax.net.ssl.trustStorePassword=changeme -Dcom.ibm.rules.authentication.oidcconfig=<ECLIPSEINITDIR>/OdmOidcProvidersRD.json
Where:
- changeme is the fixed password to be used for the default truststore.jks file.
- ECLIPSEINITDIR is the Rule Designer installation directory next to the eclipse.ini file.
-
Restart Rule Designer.
For more information, refer to this documentation.
Get hands-on experience with IBM Operational Decision Manager in a container environment by following this Getting started tutorial.
To manage ODM runtime calls, we use the Loan Validation Decision Service project
Import the Loan Validation Service in Decision Center connected as John Doe.
Deploy the Loan Validation Service production_deployment ruleapps using the production deployment deployment configuration in the Deployments>Configurations tab.
You can retrieve the payload.json from the ODM Decision Server Console or use the provided payload.
As explained in the ODM on Certified Kubernetes documentation Configuring user access with OpenID, we advise you to use basic authentication for the ODM runtime call for better performance and to avoid token expiration and revocation.
You perform a basic authentication ODM runtime call in the following way:
curl -H "Content-Type: application/json" -k --data @payload.json \
-H "Authorization: Basic b2RtQWRtaW46b2RtQWRtaW4=" \
https://<DS_RUNTIME_HOST>/DecisionService/rest/production_deployment/1.0/loan_validation_production/1.0
Where b2RtQWRtaW46b2RtQWRtaW4=
is the base64 encoding of the current username:password odmAdmin:odmAdmin
If you want to perform a bearer authentication ODM runtime call using the Client Credentials flow, you must get a bearer access token:
curl -k -X POST -H "Content-Type: application/x-www-form-urlencoded" \
-d 'client_id=<CLIENT_ID>&scope=openid&client_secret=<CLIENT_SECRET>&grant_type=client_credentials' \
'<KEYCLOAK_SERVER_URL>/protocol/openid-connect/token'
And use the retrieved access token in the following way:
curl -H "Content-Type: application/json" -k --data @payload.json \
-H "Authorization: Bearer <ACCESS_TOKEN>" \
https://<DS_RUNTIME_HOST>/DecisionService/rest/production_deployment/1.0/loan_validation_production/1.0
If you encounter any issue, have a look at the common troubleshooting explanation