ThinkPHP6使用JWT+中間件實(shí)現(xiàn)Token驗(yàn)證實(shí)例詳解
前言
最近看了很多關(guān)于TP6使用JWT的文章,總結(jié):按照他們的步驟——幾乎不行,所以準(zhǔn)備自己寫(xiě)一篇偏向?qū)崙?zhàn)的文章,也當(dāng)做個(gè)記錄。
一、JWT介紹
不喜歡搬文章,所以這篇文章,我愿稱(chēng)他為全網(wǎng)最詳!(里面詳細(xì)介紹了JWT是什么?為什么要用?優(yōu)勢(shì)、結(jié)構(gòu)、用法等)
如果想直接看代碼,請(qǐng)繼續(xù)向下↓
二、使用composer安裝JWT擴(kuò)展包
composer require firebase/php-jwt
三、在ThinkPHP6中直接使用JWT生成驗(yàn)證Token(簡(jiǎn)單粗暴)
(一)代碼文件
(已開(kāi)啟多應(yīng)用模式)
composer require topthink/think-multi-app
common.php:
<?php // 應(yīng)用公共文件 use \Firebase\JWT\JWT; use Firebase\JWT\Key;//在這個(gè)類(lèi)的參數(shù)這里踩了坑,可惡 //生成驗(yàn)簽 function signToken($data) :string { $key='LAL@lc!'; //這里是自定義的一個(gè)隨機(jī)字串,應(yīng)該寫(xiě)在config文件中的,解密時(shí)也會(huì)用,相當(dāng)于加密中常用的 鹽-salt $token=array( "iss"=>$key, //簽發(fā)者 可以為空 "aud"=>'', //面象的用戶,可以為空 "iat"=>time(), //簽發(fā)時(shí)間 "nbf"=>time()+3, //在什么時(shí)候jwt開(kāi)始生效 (這里表示生成100秒后才生效) "exp"=> time()+7200, //token 過(guò)期時(shí)間 "data"=>$data //記錄的userid的信息,這里是自已添加上去的,如果有其它信息,可以再添加數(shù)組的鍵值對(duì) ); return JWT::encode($token, $key, "HS384"); //根據(jù)參數(shù)生成了token,可選:HS256、HS384、HS512、RS256、ES256等 } //驗(yàn)證token function checkToken($token) :array { $key='LAL@lc!'; $status=array("code"=>2); try { JWT::$leeway = 60;//當(dāng)前時(shí)間減去60,把時(shí)間留點(diǎn)余地 $decoded = JWT::decode($token, new Key($key, 'HS384') ); //同上的方式,這里要和簽發(fā)的時(shí)候?qū)?yīng) $arr = (array)$decoded; $res['code']=200; $res['data']=$arr['data']; $res['data'] = json_decode(json_encode($res['data']),true);//將stdObj類(lèi)型轉(zhuǎn)換為array return $res; } catch(\Firebase\JWT\SignatureInvalidException $e) { //簽名不正確 $status['msg']="簽名不正確"; return $status; }catch(\Firebase\JWT\BeforeValidException $e) { // 簽名在某個(gè)時(shí)間點(diǎn)之后才能用 $status['msg']="token失效"; return $status; }catch(\Firebase\JWT\ExpiredException $e) { // token過(guò)期 $status['msg']="token失效"; return $status; }catch(Exception $e) { //其他錯(cuò)誤 $status['msg']="未知錯(cuò)誤"; return $status; } }
app\index\controller中:index.php
<?php namespace app\index\controller; use app\BaseController; use think\facade\Request; use think\response\Json; class Index extends BaseController { /** * 模擬將數(shù)組類(lèi)型的數(shù)據(jù)加密然后得到token * @return string */ public function index() :string { $user = ['id'=>521,'openid'=>'123456789']; $user['token'] = signToken($user); return $user['token']; } /** * 解密token的數(shù)據(jù)并返回 * @return Json */ public function checkIt() :Json { $token = Request::header()['token']??false; if(!$token) return json(['code'=>201,'msg'=>'缺少必要參數(shù):token']); $userinfo = checkToken($token); if($userinfo['code']!=200) return json(['code'=>202,'msg'=>'token驗(yàn)證失敗']); return json($userinfo); } }
(二)請(qǐng)求接口測(cè)試
1.請(qǐng)求index接口,并復(fù)制返回的token值
2.將操作1中復(fù)制的token值復(fù)制到請(qǐng)求頭中的token去
基本的使用就成功啦,接下來(lái)使用中間件來(lái)模擬登陸的情況。
四、在ThinkPHP6中使用JWT+中間件生成驗(yàn)證Token
(一)代碼文件
common.php-同上
app\index\controller中:index.php
<?php namespace app\index\controller; use app\BaseController; use think\facade\Db; use think\response\Json; class Index extends BaseController { /** * 模擬登陸接口的成功情況,將用戶關(guān)鍵信息加密到token再返回給前端 * @return Json */ public function wxLogin():Json { $userinfo = ['id'=>1,'openid'=>'123456789']; $token = ['token'=>signToken($userinfo)]; return json(['code'=>200,'msg'=>'success to login','data'=>$token]); } /** * 模擬需要驗(yàn)證token的方法做點(diǎn)事 * @return Json */ public function toDo() :Json { $user = request()->userInfo; // $userinfo = Db::name('user') // ->where('id',$user['id']) //可以使用這些信息搜索某些關(guān)鍵信息 // ->findOrEmpty(); return json(['code'=>200,'msg'=>'u can use the data to search some information or do something','data'=>$user]); } }
app\middleware中:CheckToken.php(中間件,驗(yàn)證token)
<?php namespace app\middleware; use think\Exception; use \think\facade\Request; class CheckToken { public function handle($request, \Closure $next) { try { $token = Request::header()['token']??false; if(!$token) throw new Exception('Without Token',201); $userinfo = checkToken($token); if($userinfo['code'] != 200) throw new Exception('Token checked error',202); // $userinfo['data']['token'] = $token; $request->userInfo = $userinfo['data']; } catch (Exception $err){ return json(['code'=>$err->getCode(),'msg'=>$err->getMessage()]); } return $next($request); } }
app\index\route中:app.php(路由,綁定中間件驗(yàn)證)
<?php use think\facade\Route; Route::group(function (){ //需要經(jīng)過(guò)checkToken驗(yàn)證的接口 Route::post('toDo','/toDo'); })->prefix(\app\index\controller\Index::class)->middleware(app\middleware\CheckToken::class); Route::group(function(){ //單純的路由~ Route::post('wxLogin','/wxLogin'); })->prefix(\app\index\controller\Index::class);
(二)請(qǐng)求接口測(cè)試
1.請(qǐng)求登陸接口
2.帶著登陸接口返回的token去請(qǐng)求需要驗(yàn)證token的接口 (對(duì)&錯(cuò)都試一遍)
true:(正確的嘗試)
false:(將token的第一個(gè)字符去掉)
五、總結(jié)
JWT的詳細(xì)介紹在文章開(kāi)頭的鏈接中啥都有,本文主要記錄JWT入門(mén)實(shí)戰(zhàn)的使用,知識(shí)方面互聯(lián)網(wǎng)本就存在那么多好文章~我就不多描述了。
本篇文章就到此為止辣,簡(jiǎn)單點(diǎn)說(shuō)就是用個(gè)擴(kuò)展,為啥要記錄,因?yàn)楹芏嗟奈恼路乓恍€代碼(用不得,到處報(bào)錯(cuò)),想找一篇好的文章學(xué)習(xí)也開(kāi)始變的困難起來(lái),世風(fēng)日下啊~寫(xiě)的比較匆忙,如果有錯(cuò)誤、問(wèn)題,歡迎指正提問(wèn)~
此外,JWT進(jìn)行token驗(yàn)證應(yīng)用非常廣泛,筆者測(cè)試過(guò)的node.js、Go語(yǔ)言都有JWT驗(yàn)證的相關(guān)應(yīng)用。
- Thinkphp6 配置并使用redis圖文詳解
- thinkphp6中Redis 的基本使用方法詳解
- ThinkPHP6.0前置、后置中間件區(qū)別
- ThinkPHP6.0 重寫(xiě)URL去掉Index.php的解決方法
- 基于thinkphp6.0的success、error實(shí)現(xiàn)方法
- thinkphp3.2框架集成QRcode生成二維碼的方法分析
- Thinkphp使用Zxing擴(kuò)展庫(kù)解析二維碼內(nèi)容圖文講解
- Thinkphp3.2.3整合phpqrcode生成帶logo的二維碼
- ThinkPHP6使用最新版本Endroid/QrCode生成二維碼的方法實(shí)例
相關(guān)文章
php8擴(kuò)展arginfo生成工具及工具初體驗(yàn)介紹
這篇文章主要為大家介紹了php8的擴(kuò)展arginfo生成工具及工具初體驗(yàn),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06PHP 中的面向?qū)ο缶幊蹋和ㄏ虼笮?PHP 工程的辦法
PHP 中的面向?qū)ο缶幊蹋和ㄏ虼笮?PHP 工程的辦法...2006-12-12PHP程序守護(hù)進(jìn)程化實(shí)現(xiàn)方法詳解
這篇文章主要介紹了PHP程序守護(hù)進(jìn)程化實(shí)現(xiàn)方法詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07淺析memcache啟動(dòng)以及telnet命令詳解
本篇文章是對(duì)memcache啟動(dòng)以及telnet命令進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06php實(shí)現(xiàn)監(jiān)控varnish緩存服務(wù)器的狀態(tài)
這篇文章主要介紹了php實(shí)現(xiàn)監(jiān)控varnish緩存服務(wù)器的狀態(tài),Varnish是一款高性能的開(kāi)源HTTP加速器,可以替代Squid、Nginx等服務(wù)器,需要的朋友可以參考下2014-12-12php ci 獲取表單中多個(gè)同名input元素值的代碼
這篇文章主要介紹了php ci 獲取表單中多個(gè)同名input元素值的代碼,需要的朋友可以參考下2016-03-03如何通過(guò)PHP實(shí)現(xiàn)Des加密算法代碼實(shí)例
這篇文章主要介紹了如何通過(guò)PHP實(shí)現(xiàn)Des加密算法代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05