OIDC除了授权码模式,还有隐式模式,即基于token的登录。该模式常结合移动应用或 Web App 使用。
OIDC 隐式模式不会返回授权码 code,而是直接将 access_token
和 id_token
通过 URL hash 发送到回调地址前端,后端无法获取到这里返回的值,因为 URL hash 不会被直接发送到后端。
OIDC配置
我们仍以授权码模式中提到的这张图为例:
隐式模式不需要配置App Secret
,但开启该模式后,需要保证JWKS
可用,有两种指定方式:
{
"kty": "RSA",
"e": "AQAB",
"n": "1GBBv-QOtbNIvgJZqvW2nvIrNx6-YNKJAD3L3WspAcx1y-RYctI2RBb4k4GN0du8AH2UUf8wBywONHplYAw1djkWAztHgj4cc_WxqKvD1t5bNNjRW7I5EPA9ZEkFblIAxZVwhOPK5H8KLgiVaD7y9fPEks6sVhu2VUQKC0Qr85-0WJVzmXP3QH_1yLn1qRpkJtjCW1I4DPsB0TrQC6WBMy99Io8zECraueLrJFApuRx1H_MwgDwnt4VlYuaoqU17TyBUQWO077mUB-FFI-s0jALuPAUuNWHFFogTq2cbydaSfPcWQPjylYcLcIt-bBBdedLqsTk_0nTXPqREMFwexw",
"alg": "RS256",
"use": "sig",
"kid": "AUQR2TFiVexgvm0j0PrbZ3ofEz9R2eG7qvEJP9Ua2f0"
}
其中,JWKS URL可以从服务发现地址中获取。
https://dev-5kzk7gzc.us.auth0.com/.well-known/openid-configuration
OIDC使用
获取Token
获取Token有两种方式:id_token Flow和登录接口。
区别是:id_token Flow需要依赖OIDC的登录页,而登录接口获取可以自定义登录页!
id_token Flow
以Authing为例,构建如下 id_token flow URL。
# 授权地址,与授权码模式类似
https://xxxx.authing.cn/oidc/auth?
# 客户端ID,即应用ID
client_id=644512df9e360e3f7a40e1e4&
# 重定向URL,授权完成后,跳转的页面
redirect_uri=https://example.com&
scope=openid%20profile&
# 授权类型,注意与授权码模式不同,非常重要
response_type=id_token%20token&
state=6223573295&nonce=1831289
访问后,进入登录页,输入账号密码登录。
登录成功后,将跳转到如下链接:
https://example.com/#
# 授权令牌
access_token=part1.part2.part3&
# 令牌类型,Bearer
token_type=Bearer&
# 令牌过期时间
expires_in=1209600&
scope=openid%20profile&
# id令牌
id_token=part1.part2.part3&
state=6223573295
将 access_token
保存到客户端使用,例如:local storage
。
登录接口
下面是Authing用账户密码登录的HTTP请求,详情可参考Authing的SDK。
curl --location --request POST 'https://xxxx.authing.cn/api/v3/signin' \
# 用户池ID
--header 'x-authing-userpool-id: 64452248da5930463241789f' \
# 应用ID
--header 'x-authing-app-id: 644513df9e260e3f7a10e3e4' \
--header 'Content-Type: application/json' \
--header 'Accept: */*' \
--header 'Connection: keep-alive' \
--data-raw '{
"connection":"PASSWORD", # 登录模式
"passwordPayload":{
"password":"123456", # 密码
"phone":"18189156130" # 账户
}
}'
{
"statusCode": 200,
"message": "",
"data": {
"scope": "profile openid tenant_id",
"token_type": "Bearer",
"access_token": "part1.part2.part3", // 保存使用
"expires_in": 1209600,
"id_token": "part1.part2.part3"
}
}
Fireboom官方也开源一个简单的OIDC服务,当前仅支持隐式模式,详情请查看:fb-oidc
使用Token
假设,有如下OPERATION,使用 @fromClaim
指令修饰,意思是根据当前登录用户的UID,查询待做事项列表:
query MyQuery($uid: String! @fromClaim(name: USERID)) {
todo_findManyTodo(where: {uid: {equals: $uid}}) {
id
title
}
}
其将被编译为如下REST API,请求如下:
curl -X GET "http://localhost:9991/operations/FromClaim" \
# 添加以下请求头
--header 'Authorization: Bearer <access_token>' \
-H "accept: application/json" \
{
"data": {
"todo_findUniqueClaim": {
"email": "[email protected]",
"emailVerified": false,
"location": "",
"name": "[email protected]",
"nickname": "test",
"provider": "auth0",
"roles": null,
"userId": "auth0|637b4ab0c0cb508c49de7cf3"
}
}
}
工作原理
接下来,我们学习下OIDC协议隐式模式的工作原理。
正常流程
该时序图有3个主体:客户端、Fireboom服务器、OIDC供应商 authing。
1.客户端到OIDC(Authing)
在客户端构造下述URL,跳转到Authing授权端点。
# Authing 授权端点
https://xxx.authing.cn/oidc/auth?
# 客户端ID,用于区分是哪个应用
client_id=xxxx&
# 重定向URL,当前客户端的前端页面(不是Fireboom的地址)
redirect_uri=[https://example.com]&
# 授权范围
scope=openid profile&
# 授权类型,这是与授权码模式的区别之处
response_type=id_token token&
state=xxx
2.跳转到OIDC登录页
若用户未登录,则跳转到authing的登录页,支持账户密码、手机验证码,甚至是社交登录,如微信、QQ等。
3.OIDC跳转到客户端
登录成功后,authing再跳转到客户端回调地址上,同时携带授权码access_token
和id_token
。
和授权码模式不同这里拿到的不是code,且此时未经过飞布服务。
https://example.com/#
# 授权令牌
access_token=part1.part2.part3&
# 令牌类型,Bearer
token_type=Bearer&
# 令牌过期时间
expires_in=1209600&
scope=openid%20profile&
# id令牌
id_token=part1.part2.part3&
state=6223573295
4.客户端保存令牌
客户端保存access_token
及expires_in
,供下次使用,一般存储到local storage
中。
5.客户端获取OIDC用户
客户端使用access_token
,访问OIDC的用户信息端点,获得当前用户。
curl --location --request GET 'https://xxx.authing.cn/oidc/me' \
--header 'Authorization: Bearer <access_token>' \
--header 'Accept: */*' \
{
"name": "anson",
"given_name": null,
"middle_name": null,
"family_name": null,
"nickname": "18189156130",
"preferred_username": "18189156130",
"profile": null,
"picture": null,
"website": null,
"birthdate": "2023-06-29T12:47:00.173Z",
"gender": "male",
"zoneinfo": null,
"locale": null,
"updated_at": "2023-08-09T09:07:37.988Z",
"tenant_id": "",
"sub": "645b472d9df8868fe52074d6"
}
总结一下,1-5步骤,都不需要飞布服务参与,完全可以由客户端和OIDC服务器完成。
和授权码模式最大的不同是,隐式模式不需要client secret,也不需要使用 code 换 token,更无需请求 token 端点,access_token 和 id_token 会直接从授权端点返回。
6.客户端请求Fireboom API
客户端请求Fireboom API时,携带如下请求头 ,示例见 使用Token
Authorization: Bearer <access_token>
7.Fireboom校验令牌
Fireboom 根据 配置的JWK公钥 验签access_token
。该过程不需要OIDC服务参与,因此也意味着OIDC中的Token黑名单不会生效。需要使用Fireboom的 网关钩子 自行实现黑名单。
8.客户端获取Fireboom用户
若Fireboom服务未缓存用户信息,则会用access_token
请求OIDC用户端点,获得用户,同 #5.-ke-hu-duan-huo-qu-oidc-yong-hu。
9.Fireboom调用钩子
Fireboom 调用授权钩子,在钩子中修改用户信息,返回后,由Fireboom 缓存用户信息,详情见 身份验证钩子,并返回用户信息到客户端。
简化流程
刚刚介绍的,无论是授权码模式还是隐式模式都需要使用OIDC原生的登录页。那如果想自定义登录页,又该如何做呢?
其实只要基于隐式模式稍作调整,就能支持该需求。我们知道,隐式模式的本质是获得access_token
,然后用其请求接口。
那如果OIDC服务提供了根据账户密码获得accest_token
的接口,是不是就可以供我们在自己编写的界面中直接调用了呢?
我们仍以authing为例,上述流程图就展示了该过程。
1.调用登录接口
通过post请求,输入账户、密码或手机号、验证码,获得acces_token
。不同OIDC供应商有对应实现,详情可参考其文档。例如前文中提到的 登录接口
2.客户端存储令牌
客户端存储 acces_token
,供后续使用。
其他流程没有变化,同 正常流程 的步骤6-9。
总结
总结一下,隐式模式的流程更加简洁。
由于其不依赖cookie,所以适用范围更广,不仅支持浏览器,也能支持类似微信小程序或者原生APP。
但缺点是安全性不足,因为其直接将access_token
泄露给了客户端。