Go 語言實現 QQ 掃碼登陸

pibigstar發表於2019-11-16

點選檢視 官方文件

  • appid:應用的唯一標識。在OAuth2.0認證過程中,appid的值即為oauth_consumer_key的值。
  • appkey:appid對應的金鑰,訪問使用者資源時用來驗證應用的合法性。在OAuth2.0認證過程中,appkey的值即為oauth_consumer_secret的值。

申請地址(可以用我申請的號去測試)
https://connect.qq.com/manage.html#/

QQ登入和微信登陸一樣也是採用的OAuth2.0的方式,即先獲取一個授權的code然後拿著這個code去授權中心換取token,拿著這個token就可以訪問具體的API介面了。

3.1 獲取code

func GetAuthCode(w http.ResponseWriter, r *http.Request) {
    params := url.Values{}
    params.Add("response_type", "code")
    params.Add("client_id", AppId)
    params.Add("state", "test")
    str := fmt.Sprintf("%s&redirect_uri=%s", params.Encode(), redirectURI)
    loginURL := fmt.Sprintf("%s?%s", "https://graph.qq.com/oauth2.0/authorize", str)

    http.Redirect(w, r, loginURL, http.StatusFound)
}

它會自動開啟一個網頁,我們可以點選我們目前登陸的QQ號進行登陸,或掃碼登陸

3.2 獲取access_token

當我們點選QQ登陸後,它會回撥我們後臺的地址,回撥地址的URL中會帶上授權碼code,我們根據這個code就可以獲取access_token

func GetToken(w http.ResponseWriter, r *http.Request) {
code := r.FormValue("code")
params := url.Values{}
params.Add("grant_type", "authorization_code")
params.Add("client_id", AppId)
params.Add("client_secret", AppKey)
params.Add("code", code)
str := fmt.Sprintf("%s&redirect_uri=%s", params.Encode(), redirectURI)
loginURL := fmt.Sprintf("%s?%s", "https://graph.qq.com/oauth2.0/token", str)
response, err := http.Get(loginURL)
if err != nil {
w.Write([]byte(err.Error()))
}
defer response.Body.Close()
bs, _ := ioutil.ReadAll(response.Body)
body := string(bs)
resultMap := convertToMap(body)
info := &PrivateInfo{}
info.AccessToken = resultMap["access_token"]
info.RefreshToken = resultMap["refresh_token"]
info.ExpiresIn = resultMap["expires_in"]
GetOpenId(info, w)
}

3.3 獲取OpenId

OpenId是每一個具體使用者在我們平臺下的唯一標識,後面的所有請求都會帶上這個OpenId

func GetOpenId(info *PrivateInfo, w http.ResponseWriter) {
resp, err := http.Get(fmt.Sprintf("%s?access_token=%s", "https://graph.qq.com/oauth2.0/me", info.AccessToken))
if err != nil {
w.Write([]byte(err.Error()))
}
defer resp.Body.Close()
bs, _ := ioutil.ReadAll(resp.Body)
body := string(bs)
info.OpenId = body[45:77]
GetUserInfo(info, w)
}

3.4 獲取使用者資訊

有了access_tokenopenId之後就可以去獲取使用者的資訊了

func GetUserInfo(info *PrivateInfo, w http.ResponseWriter) {
params := url.Values{}
params.Add("access_token", info.AccessToken)
params.Add("openid", info.OpenId)
params.Add("oauth_consumer_key", AppId)
uri := fmt.Sprintf("https://graph.qq.com/user/get_user_info?%s", params.Encode())
resp, err := http.Get(uri)
if err != nil {
w.Write([]byte(err.Error()))
}
defer resp.Body.Close()
bs, _ := ioutil.ReadAll(resp.Body)
w.Write(bs)
}

3.5 全部程式碼

更具體的程式碼大家可以去我的GitHub上檢視
https://github.com/pibigstar/go-demo/blob/...

為了方便大家測試與使用,AppIdAppKey 我就暫時不刪了,大家可以直接用我申請的號進行測試

相關文章