如何基于Security實(shí)現(xiàn)OIDC單點(diǎn)登錄个粱?

mark

一撵彻、說明

本文主要是給大家介紹 OIDC 的核心概念以及如何通過對(duì) Spring Security 的授權(quán)碼模式進(jìn)行擴(kuò)展來實(shí)現(xiàn) OIDC 的單點(diǎn)登錄墙贱。

OIDC 是 OpenID Connect 的簡(jiǎn)稱艺演,OIDC=(Identity, Authentication) + OAuth 2.0却紧。它在 OAuth2 上構(gòu)建了一個(gè)身份層,是一個(gè)基于 OAuth2 協(xié)議的身份認(rèn)證標(biāo)準(zhǔn)協(xié)議胎撤。我們都知道 OAuth2 是一個(gè)授權(quán)協(xié)議晓殊,它無法提供完善的身份認(rèn)證功能,OIDC 使用 OAuth2 的授權(quán)服務(wù)器來為第三方客戶端提供用戶的身份認(rèn)證伤提,并把對(duì)應(yīng)的身份認(rèn)證信息傳遞給客戶端巫俺,且完全兼容 OAuth2。

PS:理解 OIDC 的前提是需要理解 OAuth2肿男,如果對(duì) OAuth2 的單點(diǎn)登錄的原理和流程還不太了解的可以看我之前的文章《Spring Security基于Oauth2的SSO單點(diǎn)登錄怎樣做介汹?一個(gè)注解搞定

?

二、OIDC核心概念

OAuth2 提供了 Access Token 來解決授權(quán)第三方 客戶端 訪問受保護(hù)資源的問題次伶;OIDC 在這個(gè)基礎(chǔ)上提供了 ID Token 來解決第三方客戶端標(biāo)識(shí)用戶身份認(rèn)證的問題痴昧。OIDC 的核心在于 OAuth2 的授權(quán)流程中,一并提供用戶的身份認(rèn)證信息 ID Token 給到第三方 客戶端冠王,ID Token 使用 JWT 格式來包裝赶撰。

?

OIDC協(xié)議授權(quán)返回示例

{
    "resp_code": 200,
    "resp_msg": "ok",
    "datas": {
        "access_token": "d1186597-aeb4-4214-b176-08ec09b1f1ed",
        "token_type": "bearer",
        "refresh_token": "37fd65d8-f017-4b5a-9975-22b3067fb30b",
        "expires_in": 3599,
        "id_token": "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJodHRwOi8vemx0MjAwMC5jbiIsImlhdCI6MTYyMTY5NjU4MjYxNSwiZXhwIjoxNjIxNjk2NjQyNjE1LCJzdWIiOiIxIiwibmFtZSI6IueuoeeQhuWRmCIsImxvZ2luX25hbWUiOiJhZG1pbiIsInBpY3R1cmUiOiJodHRwOi8vcGtxdG1uMHAxLmJrdC5jbG91ZGRuLmNvbS_lpLTlg48ucG5nIiwiYXVkIjoiYXBwIiwibm9uY2UiOiJ0NDlicGcifQ.UhsJpHYMWRmny45K0CygXeaASFawqtP2-zgWPDnn0XiBJ6yeiNo5QAwerjf9NFP1YBxuobRUzzhkzRikWGwzramNG9na0NPi4yUQjPNZitX1JzlIA8XSq4LNsuPKO7hS1ALqqiAEHS3oUqKAsjuE-ygt0fN9iVj2LyL3-GFpql0UAFIHhew_J7yIpR14snSh3iLVTmSWNknGu2boDvyO5LWonnUjkNB3XSGD0ukI3UEEFXBJWyOD9rPqfTDOy0sTG_-9wjDEV0WbtJf4FyfO3hPu--bwtM_U0kxRbfLnOujFXyVUStiCKG45wg7iI4Du2lamPJoJCplwjHKWdPc6Zw"
    }
}

可以看到與普通的 OAuth2 相比返回的信息中除了有 access_token 之外還多出了 id_token 屬性。

?

三柱彻、什么是 ID Token

ID Token 是一個(gè)安全令牌豪娜,由授權(quán)服務(wù)器提供的包含用戶信息的 JWT 格式的數(shù)據(jù)結(jié)構(gòu),得益于 JWT(JSON Web Token)的自包含性哟楷,緊湊性以及防篡改機(jī)制瘤载,使得 ID Token 可以安全的傳遞給第三方客戶端程序并且容易被驗(yàn)證。

?

id_token包含以下內(nèi)容

{
  "iss": "http://zlt2000.cn",
  "iat": 1621696582615,
  "exp": 1621696642615,
  "sub": "1",
  "name": "管理員",
  "login_name": "admin",
  "picture": "http://xxx/頭像.png",
  "aud": "app",
  "nonce": "t49bpg"
}
  • iss:令牌頒發(fā)者
  • iat:令牌頒發(fā)時(shí)間戳
  • exp:令牌過期時(shí)間戳
  • sub:用戶id
  • name:用戶姓名
  • login_name:用戶登錄名
  • picture:用戶頭像
  • aud:令牌接收者卖擅,OAuth應(yīng)用ID
  • nonce:隨機(jī)字符串鸣奔,用來防止重放攻擊

?

3.1. 與 JWT 的 Access Token 區(qū)別

是否可以直接使用 JWT 方式的 Access Token 并在 Payload 中加入用戶信息來代替 ID Token 呢?

?

雖然在 Access Token 中可以加入用戶的信息惩阶,并且是防篡改的挎狸,但是用戶的每次請(qǐng)求都需要攜帶著 Access Token,這樣不但增加了帶寬断楷,而且很容易泄露用戶的信息锨匆。

?

3.2. 與 UserInfo 端點(diǎn)的區(qū)別

通常 OIDC 協(xié)議都需要另外提供了一個(gè) Get /userinfo 的 Endpoint,需要通過 Access Token 調(diào)用該 Endpoint 來獲取詳細(xì)的用戶信息冬筒,這個(gè)方法和 ID Token 同樣都可以獲取用戶信息恐锣,那兩者有什么區(qū)別呢茅主?

?

相比較于 Get /userinfo 的接口使用 ID Token 可以減少遠(yuǎn)程 API 調(diào)用的額外開銷;使用那個(gè)主要是看 需求土榴,當(dāng)你只需要獲取用戶的基本信息直接使用 ID Token 就可以了诀姚,并不需要每次都通過 Access Token 去調(diào)用 Get /userinfo 獲取詳細(xì)的用戶信息。

?

四鞭衩、OIDC 單點(diǎn)登錄流程

下面我們看一個(gè) OIDC 協(xié)議常用的場(chǎng)景学搜,就是具有 獨(dú)立用戶體系 系統(tǒng)間的單點(diǎn)登錄,意思指的是用戶數(shù)據(jù)并不是統(tǒng)一共用的论衍,而是每個(gè)系統(tǒng)都擁有自己獨(dú)立的用戶數(shù)據(jù),所以流程最后增加了一步 自動(dòng)注冊(cè)用戶聚磺。

file

大部分的流程與 OAuth2 的授權(quán)碼模式相同這里就不多講述了坯台,其中下面兩個(gè)步驟需要說明一下:

  • 解析 ID Token 的公鑰可以是預(yù)先提供給第三方系統(tǒng)也可以是提供接口獲取。
  • 自動(dòng)注冊(cè)用戶 指的是第一次單點(diǎn)登錄的時(shí)候瘫寝,由于用戶信息不存在需要在本系統(tǒng)中生成該用戶數(shù)據(jù)蜒蕾;例如你從未在 CSDN 中注冊(cè)也可以使用微信來登錄該網(wǎng)站。

?

五焕阿、Spring Security 實(shí)現(xiàn)

先說一下擴(kuò)展最終的目標(biāo)是需要達(dá)到以下效果:

  • 授權(quán)碼模式:/oauth/authorize?client_id={client_id}&redirect_uri={redirect_uri}&response_type=code
  • OIDC 模式:/oauth/authorize?client_id={client_id}&redirect_uri={redirect_uri}&response_type=code id_token

目標(biāo)是要通過在 response_type 中的傳值來控制是否使用 OIDC 模式咪啡,如果使用則在 response_type 中增加 id_token 的值。

由于需要在 OAuth2 返回的內(nèi)容中添加 ID Token 屬性暮屡,所以實(shí)現(xiàn)這個(gè)擴(kuò)展的關(guān)鍵就是需要通過 Security 的 TokenEnhancer 來為 Token 添加自定義字段撤摸;

定義 TokenEnhancer 的 Bean 來擴(kuò)展 Token:

file

通過授權(quán)的 response_type 參數(shù)來判斷是否需要生成 id_token。

生成 ID Token 的 JWT:

file

?

PS:上面只列出了部分關(guān)鍵代碼褒纲,完整代碼請(qǐng)通過下面的 demo 地址去下載准夷。

?

六、完整的 demo 下載地址

https://gitee.com/zlt2000/microservices-platform/tree/master/zlt-demo/sso-demo/oidc-sso

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末莺掠,一起剝皮案震驚了整個(gè)濱河市衫嵌,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌彻秆,老刑警劉巖楔绞,帶你破解...
    沈念sama閱讀 211,194評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異唇兑,居然都是意外死亡酒朵,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門幔亥,熙熙樓的掌柜王于貴愁眉苦臉地迎上來耻讽,“玉大人,你說我怎么就攤上這事帕棉≌敕剩” “怎么了饼记?”我有些...
    開封第一講書人閱讀 156,780評(píng)論 0 346
  • 文/不壞的土叔 我叫張陵,是天一觀的道長慰枕。 經(jīng)常有香客問我具则,道長,這世上最難降的妖魔是什么具帮? 我笑而不...
    開封第一講書人閱讀 56,388評(píng)論 1 283
  • 正文 為了忘掉前任博肋,我火速辦了婚禮,結(jié)果婚禮上蜂厅,老公的妹妹穿的比我還像新娘匪凡。我一直安慰自己,他們只是感情好掘猿,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,430評(píng)論 5 384
  • 文/花漫 我一把揭開白布病游。 她就那樣靜靜地躺著,像睡著了一般稠通。 火紅的嫁衣襯著肌膚如雪衬衬。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,764評(píng)論 1 290
  • 那天改橘,我揣著相機(jī)與錄音滋尉,去河邊找鬼。 笑死飞主,一個(gè)胖子當(dāng)著我的面吹牛狮惜,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播既棺,決...
    沈念sama閱讀 38,907評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼讽挟,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了丸冕?” 一聲冷哼從身側(cè)響起耽梅,我...
    開封第一講書人閱讀 37,679評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎胖烛,沒想到半個(gè)月后眼姐,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,122評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡佩番,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,459評(píng)論 2 325
  • 正文 我和宋清朗相戀三年众旗,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片趟畏。...
    茶點(diǎn)故事閱讀 38,605評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡贡歧,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情利朵,我是刑警寧澤律想,帶...
    沈念sama閱讀 34,270評(píng)論 4 329
  • 正文 年R本政府宣布,位于F島的核電站绍弟,受9級(jí)特大地震影響技即,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜樟遣,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,867評(píng)論 3 312
  • 文/蒙蒙 一而叼、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧豹悬,春花似錦葵陵、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,734評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至涤久,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間忍弛,已是汗流浹背响迂。 一陣腳步聲響...
    開封第一講書人閱讀 31,961評(píng)論 1 265
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留细疚,地道東北人蔗彤。 一個(gè)月前我還...
    沈念sama閱讀 46,297評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像疯兼,于是被迫代替她去往敵國和親然遏。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容