Keycloak
Keycloak是一个开源软件产品,旨在为现代的应用程序和服务,提供包含身份管理和访问管理功能的单点登录工具。截至2018年3月,红帽公司负责管理这一JBoss社区项目,并将其作为他们RH-SSO产品的上游项目。从概念的角度上来说,该工具的目的是,只用少量编码甚至不用编码,就能很容易地使应用程序和服务更安全。
Express + Keycloak 前端应用场景
- Resource-Based Authorization
- Resource-Based Authorization allows you to protect resources, and their specific methods/actions,* based on a set of policies defined in Keycloak, thus externalizing authorization from your application. This is achieved by exposing a
keycloak.enforcer
method which you can use to protect resources.
|
- The keycloak-enforcer method operates in two modes, depending on the value of the
response_mode
configuration option.
|
- If
response_mode
is set totoken
, permissions are obtained from the server on behalf of the subject represented by the bearer token that was sent to your application. In this case, a new access token is issued by Keycloak with the permissions granted by the server. If the server did not respond with a token with the expected permissions, the request is denied. When using this mode, you should be able to obtain the token from the request as follows:
|
- Prefer this mode when your application is using sessions and you want to cache previous decisions from the server, as well automatically handle refresh tokens. This mode is especially useful for applications acting as a client and resource server.
If
response_mode
is set topermissions
(default mode), the server only returns the list of granted permissions, without issuing a new access token. In addition to not issuing a new token, this method exposes the permissions granted by the server through therequest
as follows:
|
- Regardless of the response_mode in use, the keycloak.enforcer method will first try to check the permissions within the bearer token that was sent to your application. If the bearer token already carries the expected permissions, there is no need to interact with the server to obtain a decision. This is specially useful when your clients are capable of obtaining access tokens from the server with the expected permissions before accessing a protected resource, so they can use some capabilities provided by Keycloak Authorization Services such as incremental authorization and avoid additional requests to the server when keycloak.enforcer is enforcing access to the resource.
By default, the policy enforcer will use the client_id defined to the application (for instance, via keycloak.json) to reference a client in Keycloak that supports Keycloak Authorization Services. In this case, the client can not be public given that it is actually a resource server.
If your application is acting as both a public client(
frontend
) and resource server(backend), you can use the following configuration to reference a different client in Keycloak with the policies that you want to enforce:
|
It is recommended to use distinct clients in Keycloak to represent your
frontend
and backend.
If the application you are protecting is enabled with Keycloak authorization services and you have defined client credentials in keycloak.json, you can push additional claims to the server and make them available to your policies in order to make decisions. For that, you can define a claims configuration option which expects a function that returns a JSON with the claims you want to push:
|
Advanced authorization
readmore
根据keycloak官方指南中原文部分可以看出如何去区分keycloak的不同应用场景。
Keycloak node adapter Usage
- Instantiate a Keycloak class
The Keycloak class provides a central point for configuration and integration with your application. The simplest creation involves no arguments.
|
By default, this will locate a file named keycloak.json alongside the main executable of your application to initialize keycloak-specific settings (public key, realm name, various URLs). The keycloak.json file is obtained from the Keycloak Admin Console.
Instantiation with this method results in all of the reasonable defaults being used. As alternative, it’s also possible to provide a configuration object, rather than the keycloak.json file:
|
Applications can also redirect users to their preferred identity provider by using:
|
- Configuring a web session store
If you want to use web sessions to manage server-side state for authentication, you need to initialize the Keycloak(…) with at least a store parameter, passing in the actual session store that express-session is using.
|
- Passing a custom scope value
By default, the scope value openid is passed as a query parameter to Keycloak’s login URL, but you can add an additional custom value:
|
- Installing Middleware
Once instantiated, install the middleware into your connect-capable app:
|
- Checking Authentication
To check that a user is authenticated before accessing a resource, simply use keycloak.checkSso(). It will only authenticate if the user is already logged-in. If the user is not logged-in, the browser will be redirected back to the originally-requested URL and remain unauthenticated:
|
Protecting Resources
Simple authentication
To enforce that a user must be authenticated before accessing a resource, simply use a no-argument version of keycloak.protect():
|
Express中间件
Passport is authentication middleware for Node
passport 不能完全适配keycloak的场景,keycloak实现了自己的Adapter。但不排除未来标准化,所以保留部分passport oidc中间件策略功能,统一跳转设置未授权跳转到
oidc/login
保证逻辑完整性此外,官方文档中描述了单点登陆的特性,Adapter自动默认设置server请求logout时销毁session退出用户的实现:
Explicit user-triggered logout
By default, the middleware catches calls to /logout to send the user through a Keycloak-centric logout workflow. This can be changed by specifying a logout configuration parameter to the middleware() call:
|
说明中间件支持配置指定的退出策略,只需要配置并use即可
server端ssl等意外请求报错
- process.env.NODE_TLS_REJECT_UNAUTHORIZED = ‘0’;的配置可能会影响 Keycloak的 Node Adapter 的正常使用,此处是一个很大的坑!
readmore
- Nodejs Passport 系列之一:基础概念
Read more
Keycloak Node.js Adapter Latest 官方文档
blog-Openshift, Node and Keycloak
官方Basic NodeJS Example-keycloak-nodejs-connect
keycloak-nodejs-connect
&&keycloak Docker image
可以做一个完整的本地案例