作者:灰灰來源:JS每日一題
一、背景
傳統的web開發實現登陸功能,一般的做法是輸入賬號密碼、或者輸入手機號及短信驗證碼進行登錄
服務端校驗用戶信息通過之后,下發一個代表登錄態的 token 給客戶端,以便進行后續的交互,每當token過期,用戶都需要重新登錄
而在微信小程序中,可以通過微信官方提供的登錄能力方便地獲取微信提供的用戶身份標識,快速建立小程序內的用戶體系,從而實現登陸功能
實現小程序用戶體系主要涉及到openid和code的概念:
- 調用wx.login()方法會生成code,將code作為參數傳遞給微信服務器指定接口,就可以獲取用戶的openid
對于每個小程序,微信都會將用戶的微信ID映射出一個小程序 openid,作為這個用戶在這個小程序的唯一標識
二、流程
微信小程序登陸具體實現的邏輯如下圖所示:
- 通過 wx.login() 獲取到用戶的code判斷用戶是否授權讀取用戶信息,調用wx.getUserInfo 讀取用戶數據
- 由于小程序后臺授權域名無法授權微信的域名,所以需要自身后端調用微信服務器獲取用戶信息
- 通過 wx.request() 方法請求業務方服務器,后端把 Appid , appsecret 和 code 一起發送到微信服務器。appid 和 appsecret 都是微信提供的,可以在管理員后臺找到
- 微信服務器返回了 openid 及本次登錄的會話密鑰 session_key
- 后端從數據庫中查找 openid ,如果沒有查到記錄,說明該用戶沒有注冊,如果有記錄,則繼續往下走
- session_key 是對用戶數據進行加密簽名的密鑰。為了自身應用安全,session_key 不應該在網絡上傳輸
- 然后生成 session并返回給小程序
- 小程序把 session 存到 storage 里面
- 下次請求時,先從 storage 里面讀取,然后帶給服務端
- 服務端對比 session 對應的記錄,然后校驗有效期
更加詳細的功能圖如下所示:
三、擴展
實際業務中,我們還需要登錄態是否過期,通常的做法是在登錄態(臨時令牌)中保存有效期數據,該有效期數據應該在服務端校驗登錄態時和約定的時間(如服務端本地的系統時間或時間服務器上的標準時間)做對比
這種方法需要將本地存儲的登錄態發送到小程序的服務端,服務端判斷為無效登錄態時再返回需重新執行登錄過程的消息給小程
另一種方式可以通過調用wx.checkSession檢查微信登陸態是否過期:
如果過期,則發起完整的登錄流程
如果不過期,則繼續使用本地保存的自定義登錄態
這種方式的好處是不需要小程序服務端來參與校驗,而是在小程序端調用AP,流程如下所示:
參考文獻
https://segmentfault.com/a/1190000016750340
https://juejin.cn/post/6955754095860776973
https://www.cnblogs.com/zwh0910/p/13977278.html