一、声明:
1、本文来自:
https://packagist.org/packages/phper666/jwt-auth
2、迁移原因:
hyperf
目前在网上没什么接口认证教程,正好本人在 composer
组件库中看到该组件,认为做的比较不错,所以搬迁到 learnku
的 hyperf
社区供大家学习参考!
二、JWT-auth的特点:
特点:
jwt-auth
支持单点登录、多点登录、支持注销 token
( token
会失效)、支持刷新 token
1、单点登录:
只会有一个 token
生效,一旦刷新 token
,前面生成的 token
都会失效,一般以用户 id
来做区分
2、多点登录:
token
不做限制,一旦刷新 token
,则当前 token
会失效
**注意:使用单点登录或者多点登录时,必须要开启黑名单,并且使用 hyperf
的缓存(建议使用 redis
缓存)。如果不开启黑名单,无法使 token
失效,生成的 token
会在有效时间内都可以使用(未更换证书或者 secret
)。 **
3、单点登录原理:
JWT
有七个默认字段供选择。单点登录主要用到 jti
默认字段,jti
字段的值默认为用户 id
。当生成 token
时,getToken
方法有一个 $isInsertSsoBlack
参数来控制是否会把前面生成的 token
都失效,默认是失效的,如果想不失效,设置为 false
即可。但是如果是调用 refreshToken
来刷新 token
或者调用 logout
注销 token
,默认前面生成的 token
都会失效。
jwt
的生成的 token
加入黑名单时,会把用户 id
作为缓存的键,当前时间作为值,配置文件中的 blacklist_cache_ttl
作为缓存的失效时间。每次生成 token
或者刷新 token
时,会先从 token
中拿到签发时间和 jti
的值,根据 jti
值找到对应的缓存拿到时间,拿到时间后跟 token
的签发时间对比,如果签发时间小于等于拿到的时间值,则 token
判断为失效的。( jti
在单点登录中,存的值是用户 id
)
4、多点登录原理:
多点登录跟单点登录差不多,唯一不同的是 jti
的值不是用户 id
,而是一个唯一字符串,每次调用 refreshToken
来刷新 token
或者调用 logout
注销 token
会默认把请求头中的 token
加入到黑名单,而不会影响到别的 token
5、token不做限制原理:
token
不做限制,在 token
有效的时间内都能使用,你只要把配置文件中的 blacklist_enabled
设置为 false
即可,即为关闭黑名单功能
三、使用方法:
1、拉取依赖
composer require phper666/jwt-auth
2、发布配置
php bin/hyperf.php jwt:publish --config
3、jwt配置
去配置 config/autoload/jwt.php
文件或者在配置文件 .env
里配置
1 2 3 4
| # 务必改为你自己的字符串 JWT_SECRET=hyperf #token过期时间,单位为秒 JWT_TTL=60
|
更多的配置请到 config/autoload/jwt.php
查看
4、全局路由验证
在 config/autoload/middlewaress.php
配置文件中加入 jwt
验证中间件,所有的路由都会进行 token
的验证,例如:
1 2 3 4 5 6
| <?php return [ 'http' => [ Phper666\JwtAuth\Middleware\JwtAuthMiddleware::class ], ];
|
5、局部验证
在 config/routes.php
文件中,想要验证的路由加入 jwt
验证中间件即可,例如:
1 2 3 4 5
| <?php
Router::addGroup('/v1', function () { Router::get('/data', 'App\Controller\IndexController@getData'); }, ['middleware' => [Phper666\JwtAuth\Middleware\JwtAuthMiddleware::class]]);
|
6、注解的路由验证
请看官方文档:https://doc.hyperf.io/#/zh/middleware/middleware 在你想要验证的地方加入 jwt
验证中间件即可。
7、模拟登录获取token
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
| <?php
namespace App\Controller; use \Phper666\JwtAuth\Jwt; class IndexController extends Controller { # 模拟登录,获取token public function login(Jwt $jwt) { $username = $this->request->input('username'); $password = $this->request->input('password'); if ($username && $password) { //这里应为没有做auth的登录认证系统,为了展示随便写点数据 $userData = [ 'uid' => 1, 'username' => 'xx', ]; //获取Token $token = (string)$jwt->getToken($userData); //返回响应的json数据 return $this->response->json(['code' => 0, 'msg' => '获取token成功', 'data' => ['token' => $token]]); }
return $this->response->json(['code' => 0, 'msg' => '登录失败', 'data' => []]); }
# http头部必须携带token才能访问的路由 public function getData() { return $this->response->json(['code' => 0, 'msg' => 'success', 'data' => ['a' => 1]]); } }
|
注意:暂时不支持传入用户对象获取 token
,后期会支持
8、路由
1 2 3 4 5 6 7 8
| <?php # 登录 Router::post('/login', 'App\Controller\IndexController@login');
# 获取数据 Router::addGroup('/v1', function () { Router::get('/data', 'App\Controller\IndexController@getData'); }, ['middleware' => [Phper666\JwtAuth\Middleware\JwtAuthMiddleware::class]]);
|
9、鉴权
在需要鉴权的接口,请求该接口时在 HTTP
请求的头部加入
Authorization Bearer token
10、演示
获取 Token
将刚刚获取到的 Token
放在如下位置,请求 /v1/data
接口,如下图:
11、例子文件
路由文件:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| <?php
declare(strict_types=1); /** * This file is part of Hyperf. * * @link https://www.hyperf.io * @document https://doc.hyperf.io * @contact group@hyperf.io * @license https://github.com/hyperf-cloud/hyperf/blob/master/LICENSE */
use Hyperf\HttpServer\Router\Router;
// 登录 Router::post('/login', 'App\Controller\IndexController@login');
// 获取数据 Router::addGroup('/v1', function () { Router::get('/refresh-token', 'App\Controller\IndexController@refreshToken'); Router::get('/logout', 'App\Controller\IndexController@logout'); Router::get('/data', 'App\Controller\IndexController@getData'); }, ['middleware' => [Phper666\JwtAuth\Middleware\JwtAuthMiddleware::class]]);
|
JWT验证文件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
| <?php
declare(strict_types=1); /** * This file is part of Hyperf. * * @link https://www.hyperf.io * @document https://doc.hyperf.io * @contact group@hyperf.io * @license https://github.com/hyperf-cloud/hyperf/blob/master/LICENSE */
namespace App\Controller;
use Phper666\JwtAuth\Jwt; use Psr\Container\ContainerInterface;
class IndexController extends Controller { protected $container;
protected $jwt;
/** * 通过构造函数注入JWT. * * IndexController constructor. * * @param ContainerInterface $container * @param Jwt $jwt */ public function __construct(Jwt $jwt) { $this->jwt = $jwt; }
/** * 模拟登录. * * @return \Psr\Http\Message\ResponseInterface */ public function login() { $username = $this->request->input('username'); $password = $this->request->input('password'); if ($username && $password) { $userData = [ 'uid' => 1, // 如果使用单点登录,必须存在配置文件中的sso_key的值,一般设置为用户的id 'username' => 'xx', ]; $token = $this->jwt->getToken($userData); $data = [ 'code' => 0, 'msg' => 'success', 'data' => [ 'token' => (string) $token, 'exp' => $this->jwt->getTTL(), ], ]; return $this->response->json($data); } return $this->response->json(['code' => 0, 'msg' => '登录失败', 'data' => []]); }
/** * 刷新token,http头部必须携带token才能访问的路由. * * @throws \Psr\SimpleCache\InvalidArgumentException * @return \Psr\Http\Message\ResponseInterface */ public function refreshToken() { $token = $this->jwt->refreshToken(); $data = [ 'code' => 0, 'msg' => 'success', 'data' => [ 'token' => (string) $token, 'exp' => $this->jwt->getTTL(), ], ]; return $this->response->json($data); }
/** * 注销token,http头部必须携带token才能访问的路由. * * @throws \Psr\SimpleCache\InvalidArgumentException * @return string */ public function logout() { if ($this->jwt->logout()) { return '退出登录成功'; }; return '退出登录失败'; }
/** * http头部必须携带token才能访问的路由. * * @return \Psr\Http\Message\ResponseInterface */ public function getData() { $data = [ 'code' => 0, 'msg' => 'success', 'data' => [ 'cache_time' => $this->jwt->getTokenDynamicCacheTime(), // 获取token的有效时间,动态的 ], ]; return $this->response->json($data); } }
|
12、获取解析后的token数据
提供了一个方法 getParserData
来获取解析后的token数据。 例如:$this->jwt->getParserData()
13、建议
目前 jwt
抛出的异常目前有两种类型
Phper666\JwtAuth\Exception\TokenValidException
异常为 token
验证失败的异常,会抛出 401
Phper666\JwtAuth\Exception\JWTException,TokenValidException
JWTException
异常会抛出 500
最好自己在项目异常重新返回错误信息
四、结语
1、心得:
目前hyperf社区的生态还需要更多人去维持,希望大家能够发出更多的教程,有问题都可以在社区中提问,这样慢慢的很多问题的解决办法都能在社区中找到,社区也会越来越繁荣!
2、鸣谢:
最后感谢这个组件的开发者:phper666
他的个人博客为:https://www.liyuzhao.cn