We are investigating the potential of using Keycloak as a Single Sign-On (SSO) service for Identity and Access Management (IAM) purposes.
When using keycloak, the most common approach is to integrate the required libraries directly into the client application(s) allowing them to store tokens and include them in requests.
Although this may be as simple as adding middleware, it does require changes to the application(s).
An alternative approach is to implement edge authentication.
The example below shows nginx "at the edge" handling token introspection and making the decision whether to permit the request to proceed to the application.
Edge authentication could be especially useful for distributed systems that contain main different services that require common/centralised authentication and authorisation.
However, since the application(s) themselves do not manage sessions and tokens we will need some extra intelligence in the proxy server. Once a user authenticates this extra intelligence creates a session and sends a cookie back to the client with the response. This enables the token to accompany future requests without the client application being involved directly.
It's not clear whether the open source version of Nginx could be configured to implement cookie-based session persistence for the application(s). This, along with other capabilities like JWT validation, are available in Nginx Plus. Either way, we wouldn't want to implement this ourselves with the open source version of nginx because we could easily introduce vulnerabilities.
This linked article details the use of the lua-resty-openidc
library in the open source version of nginx, however it suggests the use of OpenResty - a modified version of nginx that enables the use of Lua which allows interpreted Lua scripts to be executed within precompiled C software like Nginx. Although it's possible to add ngx-lua yourself we wouldn't want to build out own version of nginx.
Note: OpenID Connect (openidc/OIDC) is an authentication layer built on top of OAuth 2.0 protocol making it usable through a REST API.
- The user makes a request to the app.
- Nginx, using OpenResty and lua-resty-openidc, checks if there's an existing Keycloak session cookie. If there isn't, it redirects the user to the Keycloak login page.
- The user logs in on the Keycloak page, which sets a session cookie.
- The user is redirected back to the app, this time with the session cookie (which will now be also present on future requests).
- Nginx again checks the session cookie. If the JWT validation is successful, Nginx forwards the request to the backend service.
Since Nginx/OpenResty does the JWT validation the backend service(s) don't need to be aware of Keycloak at all.
Another common alternative is to use OAuth2-proxy a popular alternative to the now discountined https://github.com/keycloak/keycloak-gatekeeper. Note gatekeeper was forked as Louketo Proxy (also discountinues) and Oneconcern Keycloak Gatekeeper.
Similar to the OpenResty implementation above, Nginx forwards the request to OAuth2-proxy which interacts with Keycloak and returns http codes informing Nginx how to proceed.
So we either install OpenResty which includes Nginx, or we add another container in addition to standard nginx for the OAuth2-proxy service...
However, neither lua-resty-openidc
nor OAuth2-proxy
are adequate solutions for fine-grained Roll-Based Access Control (RBAC) - to control access to individual API end-points. If we pursue the edge authentication architecure then in addition we also need something like Open Policy Agent (OPA) to serve as our Policy Enforcement Point (PEP).
There are other alternatives including API Gateways or Service Mesh. The open-source version of Kong is commonly used although, for RBAC, it looks like we would again need the enterprise version.
- Unless we think we need the ability to use Lua in nginx config, I'm not sure I'm comfortable with replacing nginx with OpenResty. The version of nginx bundled with OpenResty may be older, we would be relying on them for updates/security patches etc. and it feels like it increases the attack surface
- OAuth2-proxy looks like it may be an easier route than implementing OIDC in Flask middleware and would allow us to implement authentication 'on the edge'.
- Need to look into Edge Auth RBAC options in more depth. We could also confirm whether Oso would be suitable for RBAC with keycloak as this has the potential to give us increased access control across the entire Python stack including down to the database access layer/ORM (sqlalchemy-oso) but would be implemented in Flask middleware.