Golang實(shí)現(xiàn)基于時(shí)間的一次性密碼TOTP
上篇文章詳細(xì)講解了一次性密碼 OTP 相關(guān)的知識(shí),基于時(shí)間的一次性密碼 TOTP 是 OTP 的一種實(shí)現(xiàn)方式。這種方法的優(yōu)點(diǎn)是不依賴網(wǎng)絡(luò),因此即使在沒有網(wǎng)絡(luò)的情況下,用戶也可以生成密碼。所以這種方式被許多流行的網(wǎng)站使用到雙因子或多因子認(rèn)證中,包括 Google、GitHub、Facebook 和 Salesforce 等等。
因?yàn)?TOTP 是標(biāo)準(zhǔn)化的協(xié)議并且被廣泛采用,所以有很多對(duì)應(yīng)的移動(dòng)應(yīng)用或者 web 應(yīng)用實(shí)現(xiàn),被稱為身份驗(yàn)證器應(yīng)用,例如 Google Authenticator、Microsoft Authenticator 等。Golang 也有很多優(yōu)秀的三方庫可以幫助我們快速實(shí)現(xiàn) TOTP 的服務(wù)端實(shí)現(xiàn),其中比較有代表性的是 pquerna/otp 庫,接下來就使用這個(gè)庫來演示一下 TOTP 的服務(wù)端實(shí)現(xiàn)流程。
為用戶生成 TOTP Key
用戶開啟雙因子認(rèn)證時(shí),為用戶生成 TOTP Key,用于生成 TOTP 密碼。將這個(gè)密碼保存在數(shù)據(jù)庫或者秘鑰管理系統(tǒng)中,生成 key 的關(guān)鍵代碼如下:
key, err := totp.Generate(totp.GenerateOpts{ Issuer: "Github", AccountName: "user@example.com", Period: 30, Digits: otp.DigitsSix, Algorithm: otp.AlgorithmSHA1, })
這幾個(gè)參數(shù)的意思如下:
Issuer 意思是應(yīng)用名稱,例如 Github。
AccountName 意思要給哪個(gè)用戶生成 key。
Period 意思是 TOTP 密碼的有效時(shí)間,也是不同 TOTP 密碼的生成時(shí)間間隔,一般為 30 秒。
Digits 意思是生成的密碼長(zhǎng)度,一般為 6 位。
Algorithm,用于 HMAC 簽名的算法,默認(rèn)是 SHA1。
把密鑰和密碼生成規(guī)則分享給用戶
通常是將秘鑰和密碼規(guī)則信息以二維碼的形式展示給用戶,用戶使用身份驗(yàn)證器應(yīng)用掃描二維碼保存相關(guān)信息并且生成密碼。二維碼中的內(nèi)容格式一般如下:
otpauth://totp/Github:user@example.com?algorithm=SHA1&digits=6&issuer=Github&period=30&secret=5RLOAFJOB6LRV7WOKFIMDZ5IESZ7L3JM
為用戶提供“恢復(fù)碼” Recovery Codes
生成“恢復(fù)碼” Recovery Codes (使用隨機(jī)生成的字符串即可)存儲(chǔ)到數(shù)據(jù)庫或者秘鑰管理系統(tǒng)中。當(dāng)用戶不能訪問自己的 TOTP 設(shè)備(例如將 TOTP 應(yīng)用中的 TOTP 秘鑰刪除了、將 TOTP 應(yīng)用卸載了、手機(jī)丟失了等)時(shí),就無法登錄自己的帳戶了。因?yàn)檫@種情況比較常見,所以很多網(wǎng)站都會(huì)給用戶提供“備份代碼”或“恢復(fù)代碼”,并且每個(gè)只能使用一次,可以臨時(shí)用來代替 TOTP 密碼。
校驗(yàn)用戶輸入的 TOTP 密碼
用戶再次登錄后,觸發(fā)雙因子認(rèn)證,要求用戶輸入 TOTP 密碼,服務(wù)端檢驗(yàn)這個(gè)密碼。校驗(yàn)的關(guān)鍵代碼如下:
// 驗(yàn)證一次性密碼 isValid := totp.Validate(passcode, key.Secret())
模擬生成密鑰、校驗(yàn)密碼的代碼
package main import ( "fmt" "time" "github.com/pquerna/otp" "github.com/pquerna/otp/totp" ) func main() { // 生成密鑰 key, err := totp.Generate(totp.GenerateOpts{ Issuer: "Github", AccountName: "user@example.com", Period: 30, Digits: otp.DigitsSix, Algorithm: otp.AlgorithmSHA1, }) if err != nil { panic(err) } fmt.Println("Secret URL: ", key.URL()) // 模擬生成一個(gè)一次性密碼 now := time.Now() passcode, err := totp.GenerateCode(key.Secret(), now) if err != nil { panic(err) } // 驗(yàn)證一次性密碼 valid := totp.Validate(passcode, key.Secret()) if valid { fmt.Println("Valid passcode!") } else { fmt.Println("Invalid passcode!") } }
到此這篇關(guān)于Golang實(shí)現(xiàn)基于時(shí)間的一次性密碼TOTP的文章就介紹到這了,更多相關(guān)Go一次性密碼內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
一文詳解go中如何實(shí)現(xiàn)定時(shí)任務(wù)
定時(shí)任務(wù)是指按照預(yù)定的時(shí)間間隔或特定時(shí)間點(diǎn)自動(dòng)執(zhí)行的計(jì)劃任務(wù)或操作,這篇文章主要為大家詳細(xì)介紹了go中是如何實(shí)現(xiàn)定時(shí)任務(wù)的,感興趣的可以了解下2023-11-11GoFrame框架gset交差并補(bǔ)集使用實(shí)例
這篇文章主要為大家介紹了GoFrame框架gset交差并補(bǔ)集使用實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06Go基礎(chǔ)教程系列之回調(diào)函數(shù)和閉包詳解
這篇文章主要介紹了Go基礎(chǔ)教程系列之回調(diào)函數(shù)和閉包詳解,需要的朋友可以參考下2022-04-04go mayfly開源項(xiàng)目代碼結(jié)構(gòu)設(shè)計(jì)
這篇文章主要為大家介紹了go mayfly開源項(xiàng)目代碼結(jié)構(gòu)設(shè)計(jì)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11詳解Go開發(fā)Struct轉(zhuǎn)換成map兩種方式比較
本篇文章主要介紹了詳解Go開發(fā)Struct轉(zhuǎn)換成map兩種方式比較,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-07-07Go語言實(shí)現(xiàn)冒泡排序、選擇排序、快速排序及插入排序的方法
這篇文章主要介紹了Go語言實(shí)現(xiàn)冒泡排序、選擇排序、快速排序及插入排序的方法,以實(shí)例形式詳細(xì)分析了幾種常見的排序技巧與實(shí)現(xiàn)方法,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-02-02Golang中map的三種聲明定義方式實(shí)現(xiàn)
本文主要介紹了Golang中map的三種聲明定義方式實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02