当前位置:首页 > CMS教程 > Thinkphp > 列表

ThinkPHP6使用JWT+中间件实现Token验证实例详解

发布:smiling 来源: PHP粉丝网  添加日期:2023-10-25 16:54:43 浏览: 评论:0 

这篇文章主要介绍了ThinkPHP6使用JWT+中间件实现Token验证的方法,结合实例形式详细分析了JWT的功能、原理及token验证相关操作技巧,需要的朋友可以参考下,最近看了很多关于TP6使用JWT的文章,总结:按照他们的步骤——几乎不行,所以准备自己写一篇偏向实战的文章,也当做个记录。

如果想直接看代码,请继续向下↓

使用composer安装JWT扩展包

composer require firebase/php-jwt

ThinkPHP6使用JWT+中间件实现Token验证实例详解

一、在ThinkPHP6中直接使用JWT生成验证Token(简单粗暴)

(一)代码文件

(已开启多应用模式)

composer require topthink/think-multi-app

common.php:

  1. <?php 
  2. // 应用公共文件 
  3. use \Firebase\JWT\JWT; 
  4. use Firebase\JWT\Key;//在这个类的参数这里踩了坑,可恶 
  5. //生成验签 
  6. function signToken($data) :string 
  7.     $key='LAL@lc!';         //这里是自定义的一个随机字串,应该写在config文件中的,解密时也会用,相当于加密中常用的 盐-salt 
  8.     $token=array
  9.         "iss"=>$key,        //签发者 可以为空 
  10.         "aud"=>'',          //面象的用户,可以为空 
  11.         "iat"=>time(),      //签发时间 
  12.         "nbf"=>time()+3,    //在什么时候jwt开始生效  (这里表示生成100秒后才生效) 
  13.         "exp"=> time()+7200, //token 过期时间 
  14.         "data"=>$data           //记录的userid的信息,这里是自已添加上去的,如果有其它信息,可以再添加数组的键值对 
  15.     ); 
  16.     return JWT::encode($token$key"HS384");  //根据参数生成了token,可选:HS256、HS384、HS512、RS256、ES256等 
  17. //验证token 
  18. function checkToken($token) :array 
  19.     $key='LAL@lc!'
  20.     $status=array("code"=>2); 
  21.     try { 
  22.         JWT::$leeway = 60;//当前时间减去60,把时间留点余地 
  23.         $decoded = JWT::decode($tokennew Key($key'HS384') ); //同上的方式,这里要和签发的时候对应 
  24.         $arr = (array)$decoded
  25.         $res['code']=200; 
  26.         $res['data']=$arr['data']; 
  27.         $res['data'] = json_decode(json_encode($res['data']),true);//将stdObj类型转换为array 
  28.         return $res
  29.     } catch(\Firebase\JWT\SignatureInvalidException $e) { //签名不正确 
  30.         $status['msg']="签名不正确"
  31.         return $status
  32.     }catch(\Firebase\JWT\BeforeValidException $e) { // 签名在某个时间点之后才能用 
  33.         $status['msg']="token失效"
  34.         return $status
  35.     }catch(\Firebase\JWT\ExpiredException $e) { // token过期 
  36.         $status['msg']="token失效"
  37.         return $status
  38.     }catch(Exception $e) { //其他错误 
  39.         $status['msg']="未知错误"
  40.         return $status
  41.     } 

app\index\controller中:index.php

  1. <?php 
  2. namespace app\index\controller; 
  3. use app\BaseController; 
  4. use think\facade\Request; 
  5. use think\response\Json; 
  6. class Index extends BaseController 
  7.     /** 
  8.      * 模拟将数组类型的数据加密然后得到token 
  9.      * @return string 
  10.      */ 
  11.     public function index() :string 
  12.     { 
  13.         $user = ['id'=>521,'openid'=>'123456789']; 
  14.         $user['token'] = signToken($user);   
  15.         return $user['token']; 
  16.     } 
  17.     /** 
  18.      * 解密token的数据并返回 
  19.      * @return Json 
  20.      */ 
  21.     public function checkIt() :Json 
  22.     { 
  23.         $token = Request::header()['token']??false; 
  24.         if(!$token
  25.             return json(['code'=>201,'msg'=>'缺少必要参数:token']); 
  26.         $userinfo = checkToken($token); 
  27.         if($userinfo['code']!=200) 
  28.             return json(['code'=>202,'msg'=>'token验证失败']); 
  29.         return json($userinfo); 
  30.     } 

(二)请求接口测试

1.请求index接口,并复制返回的token值

ThinkPHP6使用JWT+中间件实现Token验证实例详解

2.将操作1中复制的token值复制到请求头中的token去

ThinkPHP6使用JWT+中间件实现Token验证实例详解

基本的使用就成功啦,接下来使用中间件来模拟登陆的情况。

二、在ThinkPHP6中使用JWT+中间件生成验证Token

(一)代码文件

common.php-同上

app\index\controller中:index.php

  1. <?php 
  2. namespace app\index\controller; 
  3. use app\BaseController; 
  4. use think\facade\Db; 
  5. use think\response\Json; 
  6. class Index extends BaseController 
  7.     /** 
  8.      * 模拟登陆接口的成功情况,将用户关键信息加密到token再返回给前端 
  9.      * @return Json 
  10.      */ 
  11.     public function wxLogin():Json 
  12.     { 
  13.         $userinfo = ['id'=>1,'openid'=>'123456789']; 
  14.         $token = ['token'=>signToken($userinfo)]; 
  15.         return json(['code'=>200,'msg'=>'success to login','data'=>$token]); 
  16.     } 
  17.     /** 
  18.      * 模拟需要验证token的方法做点事 
  19.      * @return Json 
  20.      */ 
  21.     public function toDo() :Json 
  22.     { 
  23.         $user = request()->userInfo; 
  24. //        $userinfo = Db::name('user') 
  25. //            ->where('id',$user['id']) //可以使用这些信息搜索某些关键信息 
  26. //            ->findOrEmpty(); 
  27.         return json(['code'=>200,'msg'=>'u can use the data to search some information or do something','data'=>$user]); 
  28.     } 

app\middleware中:CheckToken.php(中间件,验证token)

  1. <?php 
  2. namespace app\middleware; 
  3. use think\Exception; 
  4. use \think\facade\Request; 
  5. class CheckToken 
  6.     public function handle($request, \Closure $next
  7.     { 
  8.         try { 
  9.             $token = Request::header()['token']??false; 
  10.             if(!$token
  11.                 throw new Exception('Without Token',201); 
  12.             $userinfo = checkToken($token); 
  13.             if($userinfo['code'] != 200) 
  14.                 throw new Exception('Token checked error',202); 
  15. //          $userinfo['data']['token'] = $token; 
  16.             $request->userInfo = $userinfo['data']; 
  17.         } 
  18.         catch (Exception $err){ 
  19.             return json(['code'=>$err->getCode(),'msg'=>$err->getMessage()]); 
  20.         } 
  21.         return $next($request); 
  22.     } 

app\index\route中:app.php(路由,绑定中间件验证)

  1. <?php 
  2. use think\facade\Route; 
  3. Route::group(function (){    //需要经过checkToken验证的接口 
  4.     Route::post('toDo','/toDo'); 
  5. })->prefix(\app\index\controller\Index::class)->middleware(app\middleware\CheckToken::class); 
  6. Route::group(function(){    //单纯的路由~ 
  7.     Route::post('wxLogin','/wxLogin'); 
  8. })->prefix(\app\index\controller\Index::class); 

(二)请求接口测试

1.请求登陆接口

ThinkPHP6使用JWT+中间件实现Token验证实例详解

2.带着登陆接口返回的token去请求需要验证token的接口 (对&错都试一遍)

true:(正确的尝试)

ThinkPHP6使用JWT+中间件实现Token验证实例详解

false:(将token的第一个字符去掉)

ThinkPHP6使用JWT+中间件实现Token验证实例详解

三、总结

JWT的详细介绍在文章开头的链接中啥都有,本文主要记录JWT入门实战的使用,知识方面互联网本就存在那么多好文章~我就不多描述了。

本篇文章就到此为止辣,简单点说就是用个扩展,为啥要记录,因为很多的文章放一些烂代码(用不得,到处报错),想找一篇好的文章学习也开始变的困难起来,世风日下啊~写的比较匆忙,如果有错误、问题,欢迎指正提问~

此外,JWT进行token验证应用非常广泛,笔者测试过的node.js、Go语言都有JWT验证的相关应用。

Tags: JWT中间件 Token验证实例

分享到: