Sign in With Google之服務(wù)端驗(yàn)證

前言

關(guān)于使用Google第三方登錄服務(wù)端如何進(jìn)行登錄驗(yàn)證。Google登錄流程中,服務(wù)器的主要工作為驗(yàn)證用戶信息以確保此次登錄為有效登錄,然后讓當(dāng)前登錄的用戶進(jìn)行應(yīng)用服務(wù)器的相應(yīng)流程。另一方面应役,用戶在授權(quán)登錄后,應(yīng)用服務(wù)器可以通過(guò)用戶的授權(quán)獲取到access_token燥筷,以便后續(xù)訪問(wèn)Google的其他API扛吞。

登錄流程

根據(jù)上面的圖所示:

  1. 在用戶點(diǎn)擊"Sign in With Google"的按鈕后,客戶端將會(huì)收到一個(gè)來(lái)自O(shè)Auth服務(wù)器并且只能使用一次的code荆责,客戶端將該code發(fā)送至應(yīng)用服務(wù)器,這里由客戶端處理亚脆。

  2. 應(yīng)用服務(wù)器用該code向Google API服務(wù)器請(qǐng)求access_token和refresh_token做院,這里需要注意:

a. 每個(gè)code只能使用一次
b. 只要code有效,那么每次請(qǐng)求Google都將返回access_token
c. 在獲取到refresh_token之后濒持,需要將其保存起來(lái)用于后續(xù)刷新access_token使用(access_token存在過(guò)期時(shí)間)键耕,因?yàn)樵诤罄m(xù)的refresh_token交換請(qǐng)求將返回空

請(qǐng)求access_token和refresh_token的API為https://oauth2.googleapis.com/token
請(qǐng)求參數(shù)有:

參數(shù)名 解釋
code 用戶授權(quán)后客戶端獲取到的code
client_id 應(yīng)用ID
client_secret 應(yīng)用密鑰
redirect_uri API控制臺(tái)頁(yè)面 配置的授權(quán)重定向URI
grant_type 固定值:authorization_code

Google返回的參數(shù)有:

參數(shù)名 解釋
access_token 用于訪問(wèn)Google API的token
expires_in access_token有效期, 單位: 秒
id_token Google數(shù)字簽名了的用戶身份信息
scope access_token授予的訪問(wèn)范圍,以空格分隔的柑营、區(qū)分大小寫(xiě)的字符串列表表示屈雄。
token_type token類型,這里總是返回Bearer
refresh_token 用于刷新access_token的token

下面以golang為例, 發(fā)送請(qǐng)求:

    var (
        clientId     = "your clientId"
        clientSecret = "your clientSecret"
        code         = "your code"
        redirectUri  = "your redirectUri"
        url          = "https://oauth2.googleapis.com/token"
    )
    
        reader := strings.NewReader(fmt.Sprintf("client_id=%s&client_secret=%s&redirect_uri=%s&grant_type=authorization_code&code=%s", clientId, clientSecret, redirectUri, code))
        response, err := http.Post(url, "application/x-www-form-urlencoded", reader)
        handleResponse(response)
        handleErr(err)

刷新access_token的API為: https://oauth2.googleapis.com/token

請(qǐng)求參數(shù)有:

參數(shù)名 解釋
client_id 應(yīng)用ID
client_secret 應(yīng)用密鑰
refresh_token access_token值
grant_type 固定值:refresh_token

返回的參數(shù)有:

參數(shù)名 解釋
access_token 用于訪問(wèn)Google API的token
expires_in access_token有效期
token_type token類型官套,這里總是返回Bearer
scope access_token授予的訪問(wèn)范圍酒奶,以空格分隔的、區(qū)分大小寫(xiě)的字符串列表表示奶赔。
  1. 服務(wù)器獲取到access_token和refresh_token之后便可以開(kāi)始驗(yàn)證用戶信息(idToken)是否有效惋嚎。

如何驗(yàn)證用戶身份信息

用戶的身份信息是包含在idToken中的,idTokenJWT 格式站刑。按照 Google官網(wǎng) 介紹另伍,idToken需要驗(yàn)證的內(nèi)容有:

  1. 使用Google公鑰確認(rèn)idToken被正確的簽名
  2. aud的值應(yīng)該與你應(yīng)用的client_id相同
  3. iss的值為accounts.google.com或者https://accounts.google.com
  4. 有效期尚未過(guò)

官網(wǎng)建議的驗(yàn)證方式有兩種:

  1. 使用Google的驗(yàn)證庫(kù),包括Java, Node.js, PHP, Python, 當(dāng)然绞旅, Go語(yǔ)言的庫(kù)為: Golang, 如果使用Go語(yǔ)言的庫(kù)摆尝,需要注意如果你的項(xiàng)目中引入了etcd,那么你需要注意關(guān)于etcd GRPC版本所引起的一個(gè)老生常談的問(wèn)題了因悲,我在使用過(guò)程中解決辦法是降低了Golang的版本堕汞,對(duì)應(yīng)go.mod中為:
replace (
    google.golang.org/grpc => google.golang.org/grpc v1.26.0
    google.golang.org/api => google.golang.org/api v0.14.0
)

關(guān)于驗(yàn)證idToken的代碼為:

import (
    "http"
    "github.com/dghubble/oauth1"
    "google.golang.org/api/option"
    "google.golang.org/api/oauth2/v2"
)
    
    oatuService, err := oauth2.NewService(context.Background(), option.WithHTTPClient(http.DefaultClient))
    if err != nil {
        log4go.Errorf("%v\n", err)
        return nil, err
    }
    tokenInfoCall := oatuService.Tokeninfo()
    tokenInfoCall.IdToken(googleToken)
    tokenInfo, err := tokenInfoCall.Do()
    //根據(jù)tokenInfo中的字段進(jìn)行驗(yàn)證

注意:Google建議生產(chǎn)環(huán)境使用此方法進(jìn)行驗(yàn)證

  1. 調(diào)用Google的API進(jìn)行驗(yàn)證, 驗(yàn)證API為: https://oauth2.googleapis.com/tokeninfo?id_token={idToken}, 如:

https://oauth2.googleapis.com/tokeninfo?id_token=XYZ123

請(qǐng)求方法既可以是POST也可以是GET, 需要注意的是, 在收到API回復(fù)后只有當(dāng)StatusCode為http.StatusOK時(shí)才能進(jìn)行下一步校驗(yàn)。

idToken校驗(yàn)API返回的結(jié)果為JSON鍵值對(duì)晃琳,如:

{
 // 這6個(gè)字段是所有idToken都包含的
 "iss": "https://accounts.google.com",  //token簽發(fā)者臼朗,值為https://accounts.google.com或者accounts.google.com
 "sub": "110169484474386276334", //用戶在該Google應(yīng)用中的唯一標(biāo)識(shí)邻寿,類似于微信的OpenID
 "azp": "1008719970978-hb24n2dstb40o45d4feuo2ukqmcc6381.apps.googleusercontent.com", //具體我也不知道,猜測(cè)與aud相同视哑,都是應(yīng)用的client_id
 "aud": "1008719970978-hb24n2dstb40o45d4feuo2ukqmcc6381.apps.googleusercontent.com", //client_id
 "iat": "1433978353", //簽發(fā)時(shí)間
 "exp": "1433981953", //過(guò)期時(shí)間

 // 下面的字段只有當(dāng)用戶授權(quán)了"profile"和"email"權(quán)限后才會(huì)出現(xiàn)
 "email": "testuser@gmail.com", //用戶郵箱
 "email_verified": "true", //郵箱是否已驗(yàn)證
 "name" : "Test User", //用戶名
 "picture": "https://lh4.googleusercontent.com/-kYgzyAWpZzJ/ABCDEFGHI/AAAJKLMNOP/tIXL9Ir44LE/s99-c/photo.jpg", //用戶頭像
 "given_name": "Test", //名
 "family_name": "User", //姓
 "locale": "en"  //所屬地區(qū)
}

注意: 因?yàn)樵谑跈?quán)過(guò)程中和獲取access_token的過(guò)程中都會(huì)獲取到idToken绣否,所以服務(wù)器驗(yàn)證的idToken既可以由客戶端傳給服務(wù)器,也可以是服務(wù)器自己獲取挡毅。但是在登錄過(guò)程中蒜撮,客戶端獲取到的授權(quán)code必須傳給服務(wù)器

參考資料

Google Sign-In for server-side apps
Authenticate with a backend server
Using OAuth 2.0 to Access Google APIs

原文鏈接

Sign in With Google之服務(wù)端驗(yàn)證

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市跪呈,隨后出現(xiàn)的幾起案子段磨,更是在濱河造成了極大的恐慌,老刑警劉巖耗绿,帶你破解...
    沈念sama閱讀 217,542評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件苹支,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡误阻,警方通過(guò)查閱死者的電腦和手機(jī)债蜜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,822評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)究反,“玉大人寻定,你說(shuō)我怎么就攤上這事【停” “怎么了狼速?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,912評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)卦停。 經(jīng)常有香客問(wèn)我向胡,道長(zhǎng),這世上最難降的妖魔是什么惊完? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,449評(píng)論 1 293
  • 正文 為了忘掉前任捷枯,我火速辦了婚禮,結(jié)果婚禮上专执,老公的妹妹穿的比我還像新娘淮捆。我一直安慰自己,他們只是感情好本股,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,500評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布攀痊。 她就那樣靜靜地躺著,像睡著了一般拄显。 火紅的嫁衣襯著肌膚如雪苟径。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,370評(píng)論 1 302
  • 那天躬审,我揣著相機(jī)與錄音棘街,去河邊找鬼蟆盐。 笑死,一個(gè)胖子當(dāng)著我的面吹牛遭殉,可吹牛的內(nèi)容都是我干的石挂。 我是一名探鬼主播,決...
    沈念sama閱讀 40,193評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼险污,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼痹愚!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起蛔糯,我...
    開(kāi)封第一講書(shū)人閱讀 39,074評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤拯腮,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后蚁飒,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體动壤,經(jīng)...
    沈念sama閱讀 45,505評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,722評(píng)論 3 335
  • 正文 我和宋清朗相戀三年淮逻,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了琼懊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,841評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡弦蹂,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出强窖,到底是詐尸還是另有隱情凸椿,我是刑警寧澤,帶...
    沈念sama閱讀 35,569評(píng)論 5 345
  • 正文 年R本政府宣布翅溺,位于F島的核電站脑漫,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏咙崎。R本人自食惡果不足惜优幸,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,168評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望褪猛。 院中可真熱鬧网杆,春花似錦、人聲如沸伊滋。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,783評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)笑旺。三九已至昼浦,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間筒主,已是汗流浹背关噪。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,918評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工鸟蟹, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人使兔。 一個(gè)月前我還...
    沈念sama閱讀 47,962評(píng)論 2 370
  • 正文 我出身青樓建钥,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親火诸。 傳聞我的和親對(duì)象是個(gè)殘疾皇子锦针,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,781評(píng)論 2 354

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