Instead of setting up multiple realms or multiple client registrations, you can make use of a single realm and a single client registration and still easily support multi tenancy using Keycloak. It will be more secure to have e.g. isolated realms, as users then are separated, but I'm building a simple app that will have a single install and single configured client app and data store and in need for supporting a user to belong to multiple tenants. This is what this post is about. It's only four steps.
Steps
- Add User attributes using a key-convention.
- Add a Client scope that will represent tenants.
- Add a mapper to extract the User attributes.
- Add Client scope to the Client in use.
Step 1 - Add User attributes using a key-convention
You are probably using Keycloaks API for this via an integration, but for ease, I'll use the Keycloak GUI.
data:image/s3,"s3://crabby-images/f0b47/f0b47af93ca6edb11358e18bf07888d6c948444d" alt=""
Step 2 - Add a Client scope that will represent tenants
Create a new Client scope with naming of your choice. I usually add a short prefix for the app, e.g. "foo-"
data:image/s3,"s3://crabby-images/ce423/ce423bf917df612980c4053f728bf73e10fa015a" alt=""
Step 3 - Add a mapper to extract the User attributes
Now we need to add contents to our created Client scope. We will do this by configuring a new mapper.
data:image/s3,"s3://crabby-images/bb10f/bb10fbd3f1ec1e7603cb9914db5575efaedb125f" alt=""
Select an "User attribute"
mapper.
data:image/s3,"s3://crabby-images/9bebe/9bebef69b0b09ac925c5eab8f6d99f1b40ae2c8c" alt=""
Configure the user attribute to extract from the user account and if you have the same case as I do, in the need of belonging to multiple tenants, ensure it's multivalued.
data:image/s3,"s3://crabby-images/a0e06/a0e062c9f267e7a627cc89a275e8cd8db662b69d" alt=""
Step 4 - Add Client scope to the Client in use
The final step is to register the newly created Client scope with your client.
data:image/s3,"s3://crabby-images/24f20/24f20c1cfafc172ae42aa68b4049757661a8b0fd" alt=""
The Result
Login with the user and look at the scopes for the access token.
data:image/s3,"s3://crabby-images/0bf87/0bf87da9b686583123bae858850bf619c9e68d56" alt=""
The mapped scope is now present in the parsed token:
data:image/s3,"s3://crabby-images/3ce5e/3ce5eea5b5af21e2f987e4ec40824160252e27a1" alt=""
foo-tenants: ["Tenant1", "Tenant2"]
email_verified: false
name: Tester Testersson
preferred_username: tester
given_name: Tester
family_name: Testersson
That's it. Happy coding!
//Daniel