```
~~~
<?php
namespace app\program\controller; //命名空間
use think\App;
use think\facade\Db;
class ProgramService //類名(可自定義)
{
private $table = "數據庫表名"; //我的數據庫表名 取你自己的表名或者表名寫在SQL語句中
~~~
~~~
private $authorization_uri = "https://api.weixin.qq.com/sns/jscode2session"; //微信授權獲取openID以及session_key的地址
//自定義變量
private $AppID;
private $Secret;
private $session_key;
private $openid;
private $unionid;
public function __construct()
{
//構造方法,用于變量賦值 也可以在定義變量時直接賦值 注, 類變量不能直接調用方法
$this -> AppID = sysconf("program.app_id");
$this -> Secret = sysconf("program.secret");
}
/**
* 小程序授權入口
* 接收小程序傳過來的code值獲取openID及sessionkey
* @param $code
* @param $rawData
* @param $session_key
* @param $signature
* @param $iv
* @param $encryptedData
* @return \think\response\Json
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function authorization($code,$rawData,$signature,$iv,$encryptedData,$from_id = ''){
if(empty($code)) return json(returnData(1,'小程序授權參數code不存在'));
//拼接url地址
$url = $this -> authorization_uri . "?appid=" . $this -> AppID . "&secret=" . $this -> Secret . "&js_code=" . $code . "&grant_type=authorization_code";
//自己封裝的請求 composer 安裝php-curl 直接調用curl 請求即可 封裝是為了簡化代碼
$res = json_decode(requestGetData($url),true);
// dump($res);die;
if(isset($res['errcode'])) return json(returnData(1,'授權失敗',$res['errmsg']));
//變量賦值
$this -> session_key = $res['session_key'];
$this -> openid = $res['openid'];
// $this -> unionid = $res['unionid'];
//驗證簽名。判斷數據真實性
$signature2 = sha1(htmlspecialchars_decode($rawData) . $this -> session_key);
if($signature != $signature2) return json(returnData(1,'授權失敗',"簽名認證失敗"));
//檢查用戶登錄狀態
return $this -> changeLoginStatus($this -> session_key,$iv,$encryptedData,$from_id);
}
/**
* 檢查用戶登錄狀態
* @param $session_key
* @param $iv
* @param $encryptedData
* @return \think\response\Json
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function changeLoginStatus($session_key,$iv,$encryptedData,$from_id){
//根據openID判斷數據庫是否存在該用戶。存在即直接返回用戶數據
$user = Db::name($this -> table) -> where("openid",$this -> openid) -> find();
if($user) {
//是否被人邀請 如無需此功能可刪除
if($from_id != '' && $user['from_id'] == 0){
Db::name($this -> table) -> where("id",$user['id']) -> update(['form_id' => $from_id]);
}
return json(returnData(99,'登錄成功',$user['openid']));
}
//用戶未授權,調用微信授權解密參數并存入數據庫
return $result = $this -> decryptionEncryptedData($session_key,$iv,$encryptedData,$from_id);
}
/**
* 解密參數,保存用戶數據
* @param $session_key
* @param $iv
* @param $encryptedData
* @return \think\response\Json
*/
public function decryptionEncryptedData($session_key,$iv,$encryptedData,$from_id){
if (strlen($session_key) != 24) return json(returnData(1,'授權失敗',"解密后得到的sessionKey非法"));
if (strlen($iv) != 24) return json(returnData(1,'授權失敗',"解密后得到的iv非法"));
$aesKey = base64_decode($session_key);
$aesIV = base64_decode($iv);
$aesCipher = base64_decode($encryptedData);
$result = openssl_decrypt( $aesCipher, "AES-128-CBC", $aesKey, 1, $aesIV);
$result = json_decode($result,true);
// dump($result);die;
if(empty($result)) return json(returnData(1,'授權失敗',"解密后得到的Buffer非法"));
//確認小程序AppID是否為同一個
if($result['watermark']['appid'] != $this -> AppID ) return json(returnData(1,'授權失敗',"解密后得到的AppID非法"));
//需保存的用戶數組 請改成你自己的數據庫字段
$customer['openid'] = $this -> openid;
//開放平臺標識,如無需開放平臺 請注釋或刪除
$customer['unionid'] = $result['unionId'];
//用戶昵稱可能包含特殊字符,需處理
$customer['nickname'] = $this->removeEmoji($result['nickName']);
$customer['sex'] = $result['gender'];
$customer['city'] = $result['city'];
$customer['level'] = $this -> setCustomerLevel();
$customer['from_id'] = $from_id;
$customer['province'] = $result['province'];
$customer['country'] = $result['country'];
$customer['headimgurl'] = $result['avatarUrl'];
$customer['appid'] = $result['watermark']['appid'];
$customer['authorization_time'] = $result['watermark']['timestamp'];
$id = Db::name($this -> table) -> insertGetId($customer);
if($id) return json(returnData(99,'授權成功',$this -> openid));
return json(returnData(1,'授權失敗',"保存用戶信息出錯".Db::name($this -> table) -> getLastSql()));
}
/**
* 處理特殊字符
* @param $text
* @return string|string[]|null
*/
public function removeEmoji($text) {
$clean_text = "";
preg_match_all("/[\x{4e00}-\x{9fa5}|0-9|a-z|A-Z|_]/u", $text, $matches);
$clean_text = isset($matches[0]) ? implode('', $matches[0]) : '';
// Match Emoticons
$regexEmoticons = '/[\x{1F600}-\x{1F64F}]/u';
$clean_text = preg_replace($regexEmoticons, '', $text);
// Match Miscellaneous Symbols and Pictographs
$regexSymbols = '/[\x{1F300}-\x{1F5FF}]/u';
$clean_text = preg_replace($regexSymbols, '', $clean_text);
// Match Transport And Map Symbols
$regexTransport = '/[\x{1F680}-\x{1F6FF}]/u';
$clean_text = preg_replace($regexTransport, '', $clean_text);
// Match Miscellaneous Symbols
$regexMisc = '/[\x{2600}-\x{26FF}]/u';
$clean_text = preg_replace($regexMisc, '', $clean_text);
// Match Dingbats
$regexDingbats = '/[\x{2700}-\x{27BF}]/u';
$clean_text = preg_replace($regexDingbats, '', $clean_text);
return $clean_text;
}
?>
~~~
```
- thinkphp6執行流程(一)
- php中use關鍵字用法詳解
- Thinkphp6使用騰訊云發送短信步驟
- 路由配置
- Thinkphp6,static靜態資源訪問路徑問題
- ThinkPHP6.0+ 使用Redis 原始用法
- smarty在thinkphp6.0中的最佳實踐
- Thinkphp6.0 搜索器使用方法
- 從已有安裝包(vendor)恢復 composer.json
- tp6with的用法,表間關聯查詢
- thinkphp6.x多對多如何添加中間表限制條件
- thinkphp6 安裝JWT
- 緩存類型
- 請求信息和HTTP頭信息
- 模型事件用法
- 助手函數匯總
- tp6集成Alipay 手機和電腦端支付的方法
- thinkphp6使用jwt
- 6.0session cookie cache
- tp6筆記
- TP6(thinkphp6)隊列與延時隊列
- thinkphp6 command(自定義指令)
- command(自定義指令)
- 本地文件上傳
- 緩存
- 響應
- 公共函數配置
- 七牛云+文件上傳
- thinkphp6:訪問多個redis數據源(thinkphp6.0.5 / php 7.4.9)
- 富文本編輯器wangEditor3
- IP黑名單
- 增刪改查 +文件上傳
- workerman 定時器操作控制器的方法
- 上傳文件到阿里云oss
- 短信或者郵箱驗證碼防刷代碼
- thinkphp6:訪問redis6(thinkphp 6.0.9/php 8.0.14)
- 實現關聯多個id以逗號分開查詢數據
- thinkphp6實現郵箱注冊功能的細節和代碼(點擊鏈接激活方式)
- 用mpdf生成pdf文件(php 8.1.1 / thinkphp v6.0.10LTS )
- 生成帶logo的二維碼(php 8.1.1 / thinkphp v6.0.10LTS )
- mysql數據庫使用事務(php 8.1.1 / thinkphp v6.0.10LTS)
- 一,創建過濾IP的中間件
- 源碼解析請求流程
- 驗證碼生成
- 權限管理
- 自定義異常類
- 事件監聽event-listene
- 安裝與使用think-addons
- 事件與多應用
- Workerman 基本使用
- 查詢用戶列表按拼音字母排序
- 擴展包合集
- 查詢用戶數據,但是可以通過輸入用戶昵稱來搜索用戶同時還要統計用戶的文章和粉絲數
- 根據圖片的minetype類型獲取文件真實拓展名思路
- 到處excel
- 用imagemagick庫生成縮略圖
- 生成zip壓縮包并下載
- API 多版本控制
- 用redis+lua做限流(php 8.1.1 / thinkphp v6.0.10LTS )
- 【thinkphp6源碼分析三】 APP類之父, 容器Container類
- thinkphp6表單重復提交解決辦法
- 小程序授權
- 最簡單的thinkphp6導出Excel
- 根據訪問設備不同訪問不同模塊
- 服務系統
- 前置/后置中間件
- 給接口api做簽名驗證(php 8.1.1 / thinkphp v6.0.10LTS )
- 6實現郵箱注冊功能的細節和代碼(點擊鏈接激活方式)
- 使用前后端分離的驗證碼(thinkphp 6.0.9/php 8.0.14/vue 3.2.26)
- 前后端分離:用jwt+middleware做用戶登錄驗證(php 8.1.1 / thinkphp v6.0.10LTS )
- vue前后端分離多圖上傳
- thinkphp 分組、頁面跳轉與ajax
- thinkphp6 常用方法文檔
- 手冊里沒有的一些用法
- Swagger 3 API 注釋
- PHP 秒級定時任務
- thinkphp6集成gatewayWorker(workerman)實現實時監聽
- thinkphp6按月新增數據表
- 使用redis 實現消息隊列
- api接口 統一結果返回處理類
- 使用swoole+thinkphp6.0+redis 結合開發的登錄模塊
- 給接口api做簽名驗證
- ThinkPHP6.0 + UniApp 實現小程序的 微信登錄
- ThinkPHP6.0 + Vue + ElementUI + axios 的環境安裝到實現 CURD 操作!
- 異常$e
- 參數請求驗證自定義和異常錯誤自定義