網站實現微信登錄之回調函數中登錄邏輯的處理--基於yii2開發的描述

来源:http://www.cnblogs.com/yangtoude/archive/2016/12/18/wechat-login-callback-logic-based-on-yii2-development-experience.html
-Advertisement-
Play Games

上一篇文章網站實現微信登錄之嵌入二維碼中描述瞭如何在自己的登錄頁面內嵌入登錄二維碼,今天的這篇文章主要是描述下在掃碼成功之後微信重定向回網站後登錄邏輯的處理,其實也就是驗證身份信息,授權用戶登錄的邏輯。這裡說句題外話,寫博客複習已經做過的項目真的有助於自己對已經寫過代碼和業務邏輯的理解,說不定還有意 ...


上一篇文章網站實現微信登錄之嵌入二維碼中描述瞭如何在自己的登錄頁面內嵌入登錄二維碼,今天的這篇文章主要是描述下在掃碼成功之後微信重定向回網站後登錄邏輯的處理,其實也就是驗證身份信息,授權用戶登錄的邏輯。這裡說句題外話,寫博客複習已經做過的項目真的有助於自己對已經寫過代碼和業務邏輯的理解,說不定還有意外的收穫。所謂,“溫故而知新”,我會保持寫博客的習慣。

1,微信掃碼成功之後

在用戶掃碼成功之後,pc端網站上的二維碼會出現如下的提示:(這裡是用的微信開發文檔中的例子1號店網站用來演示效果)。

這裡需要註意的是,微信開發文檔中的例子請求登錄1號店網站,它是給出了一個微信登錄的鏈接https://passport.yhd.com/wechat/login.do,點擊登錄鏈接後會重定向到掃描二維碼的頁面,頁面鏈接如下:https://open.weixin.qq.com/connect/qrconnect?appid=wxbdc5610cc59c1631&redirect_uri=https%3A%2F%2Fpassport.yhd.com%2Fwechat%2Fcallback.do&response_type=code&scope=snsapi_login&state=e25d9455e8f8e6f60a9fef0ba6509889#wechat_redirect,這個做法與上一篇文章網站實現微信登錄之嵌入二維碼的做法不同:它沒有將二維碼用js嵌入到登錄頁面login.do中,而是直接在登錄頁面中進行了重定向:

header('Location: https://open.weixin.qq.com/connect/qrconnect?appid=wxbdc5610cc59c1631&redirect_uri=https%3A%2F%2Fpassport.yhd.com%2Fwechat%2Fcallback.do&response_type=code&scope=snsapi_login&state=3d6be0a4035d839573b04816624a415e#wechat_redirect');

在實際開發中,應該根據設計的要求或客戶需求來選擇利用js嵌入還是在頁面中重定向的方法。

然後,在手機端的微信中會彈出提示用戶確認登陸1號店的界面:

點擊登錄按鈕後,pc端網頁會跳轉到哪兒去呢?這個也要根據實際的業務邏輯來處理。

2,獲取access_token和授權用戶的openid

在redirect_uri的URL映射的對應的action中來通過code獲取access_token。

 1     public function actionCallback($code, $state)
 2     {
 3         // 獲取並校驗前臺存儲的隨機串,防csrf攻擊
 4         $session = Yii::$app->session;
 5         if ($state != $session->get('wx_state')) {
 6             exit();
 7         }
 8         $session->remove('wx_state');
 9 
10         // 微信開放平臺網站應用的appid和秘鑰secret
11         $appid = 'appid';
12         $secret = 'secret';
13 
14         $curl = new Curl();
15         $wxresponse = $curl->get('https://api.weixin.qq.com/sns/oauth2/access_token?appid=' . $appid
16             . '&secret=' . $secret . '&code=' . $code . '&grant_type=authorization_code');
17         $wxresult = json_decode($wxresponse);

註意:上面代碼第14行用到了一個yii2的curl擴展,感興趣的可以去github上看下。

第15行請求成功後返回的參數如下:

{ 
"access_token":"ACCESS_TOKEN", 
"expires_in":7200, 
"refresh_token":"REFRESH_TOKEN",
"openid":"OPENID", 
"scope":"SCOPE" 
}

 

參數	        說明
access_token	介面調用憑證
expires_in	access_token介面調用憑證超時時間,單位(秒)
refresh_token	用戶刷新access_token
openid	        授權用戶唯一標識
scope	        用戶授權的作用域,使用逗號(,)分隔

 

actionCallback方法的參數中,$code是用戶掃碼確認登錄後微信返回的,$state是自己設置的參數,詳細可見網站實現微信登錄之嵌入二維碼一文。請求得到access_token和openid最後被存儲到$wxresult中。

這裡需要註意下,此處返回的openid與第3節中返回的openid是不同的,註意看參數說明。

3,利用上面獲取的token和openid獲取用戶的個人信息,查詢資料庫驗證登錄。成功則設置session,不成功(新用戶),可以考慮添加新用戶實現掃碼註冊的邏輯。

 1         if (isset($wxresult->errcode) && $wxresult->errcode > 0) {
 2             // 向微信請求授權時出錯,列印錯誤碼
 3             echo json_encode($wxresult);
 4             exit;
 5         } else {
 6             // 獲取用戶個人信息(unionid)
 7             $response = $curl->get('https://api.weixin.qq.com/sns/userinfo?access_token=' . $wxresult->access_token
 8                 . '&openid=' . $wxresult->openid);
 9             $result = json_decode($response);
10             $wxUser = WxUser::find()->where(['wx_unionid' => $result->unionid])->one();
11             if ($wxUser) {
12                 // 登錄
13                 $result = Yii::$app->user->login($wxUser->id, 3600 * 24 * 30);
14                 if ($result) {
15                     // 登錄成功,設置session
16                     Yii::$app->session['wxuser'] = $wxUser->id;
17                 } else {
18                     // 登錄失敗
19                     echo 'login 失敗';
20                     exit();
21                 }
22             } else {
23                 // 創建新用戶
24             }
25         }

第7行請求成功後,返回的參數示例如下:

{ 
"openid":"OPENID",
"nickname":"NICKNAME",
"sex":1,
"province":"PROVINCE",
"city":"CITY",
"country":"COUNTRY",
"headimgurl": "http://wx.qlogo.cn/mmopen/g3MonUZtNHkdmzicIlibx6iaFqAc56vxLSUfpb6n5WKSYVY0ChQKkiaJSgQ1dZuTOgvLLrhJbERQQ4eMsv84eavHiaiceqxibJxCfHe/0",
"privilege":[
"PRIVILEGE1", 
"PRIVILEGE2"
],
"unionid": " o6_bmasdasdsad6_2sgVt7hMZOPfL"

}

 

參數	        說明
openid	        普通用戶的標識,對當前開發者帳號唯一
nickname	普通用戶昵稱
sex	        普通用戶性別,1為男性,2為女性
province	普通用戶個人資料填寫的省份
city	        普通用戶個人資料填寫的城市
country	        國家,如中國為CN
headimgurl	用戶頭像,最後一個數值代表正方形頭像大小(有0、46、64、96、132數值可選,0代表640*640正方形頭像),用戶沒有頭像時該項為空
privilege	用戶特權信息,json數組,如微信沃卡用戶為(chinaunicom)
unionid	        用戶統一標識。針對一個微信開放平臺帳號下的應用,同一用戶的unionid是唯一的。

 

上面代碼第10行是依據用戶的unionid是否在資料庫中來驗證其是否為網站的可登錄用戶,這裡為什麼要使用用戶的unionid呢?這裡涉及到了微信授權介面的UnionID機制。

微信開發文檔授權後介面調用(UnionID)中對於unionid的解釋:此介面用於獲取用戶個人信息。開發者可通過OpenID來獲取用戶基本信息。特別需要註意的是,如果開發者擁有多個移動應用、網站應用和公眾帳號,可通過獲取用戶基本信息中的unionid來區分用戶的唯一性,因為只要是同一個微信開放平臺帳號下的移動應用、網站應用和公眾帳號,用戶的unionid是唯一的。換句話說,同一用戶,對同一個微信開放平臺下的不同應用,unionid是相同的。

我的理解是,每個用戶對各公眾號的OpenID是唯一的,而對於不同公眾號,同一用戶的openid是不同的。那麼如果有多個公眾號(服務號、訂閱號),用openid就不能區分用戶的唯一身份,就只能用unionid了。就算只有一個公眾號,也建議採用unionid來區分用戶身份的唯一性。

 

4,小結

微信掃碼登錄或註冊方便了用戶,但卻增加了開發者的開發、調試等工作量。其實不止是開發階段的工作量,整個網站在設計時就要考慮到這個功能,登錄界面的設計、頁面的跳轉等。

 

參考:

微信開放平臺資源中心



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.Node.js 文件系統 2.非同步和同步 讀取文件內容的函數有非同步的 fs.readFile() 和同步的 fs.readFileSync()。 ...
  • R是免費開源的軟體,具有強大的數據處理和繪圖等功能。下麵是R開發環境的搭建過程。 一、點擊網址 https://www.r-project.org/ ,進入“The R Project for Statistical Computing”界面。 二、點擊加粗字體的“download R”,進行“CR ...
  • 1.事件驅動程式綁定事件及事件的處理程式 我們可以通過程式觸發事件 2.實例 3.讓類都具有事件的功能 app.js Test.js ...
  • 實現客戶端與服務端的連接: 使用ServerSocket創建TCP服務端 ServerSocket :此類實現伺服器套接字。伺服器套接字請求通過網路傳入,基於該請求執行某些操作,然後可能向請求者返回結果。 詳細描述見:http://download.java.net/jdk/jdk-api-loca ...
  • 1.什麼是模塊? 為了讓Node.js的文件可以相互調用,Node.js提供了一個簡單的模塊系統。 模塊是Node.js 應用程式的基本組成部分,文件和模塊是一一對應的。換言之,一個 Node.js 文件就是一個模塊,這個文件可能是JavaScript 代碼、JSON 或者編譯過的C/C++ 擴展。 ...
  • # -*- coding:utf-8 -*- import os def systemInfo(): #os.name 獲取系統的平臺 Windows 返回 ‘nt'; Linux 返回’posix' if os.name=='nt': print "你的系統是Windows系統" else: pr ...
  • 簡介:學習完了php和jQuery之後,對函數的記憶不到位,導致很多函數沒記住,所以為了促進自己的記憶,每天花一點時間來寫這個博客。 時間:2016-12-18 地點:太原 天氣:晴 一.php函數(數組相關的函數) 1.array_change_key_case 作用:返回字元串鍵名為全大寫或者全 ...
  • 1.class是struct的擴展,它包括數據成員和成員函數。 2.在C++中,有三種訪問許可權: (1)private:預設,只供類內部的函數使用。 (2)public:類外的程式可以使用。 (3)proteted 註意: C++的規範,類名稱的首字母應該大寫。 eg: 出現錯誤: 3.通過函數來訪 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...