写一个BaseController 类
基本思路:
1、继承一个公共基类,将验证相关代码放在基类
2、根据 孩子类下的notNeedToken 来决定是否进行验证
3、验证失败后,直接响应回来
这里需要封装一个主要代码:
php
protected function response($data = [])
{
$type = $this->getResponseType();
$response = Response::create($data, $type);
throw new HttpResponseException($response);
}
如果直接return 返回,是不会终止执行的,而是会继续执行到指定的控制器,所以这里的方法仿造了$this->error() 以及 $this->success()方法。
全部代码
php
<?php
namespace app\api\controller;
use think\Controller;
use think\exception\HttpResponseException;
use think\Request;
use think\Response;
class BaseController extends Controller
{
protected $tokenError = ''; // 错误信息
protected $tokenInfo = null; // 解析出来的token信息
protected $statusCode = 200;
protected $notNeedToken = [];
public function _initialize()
{
$this->checkToken();
$this->statusCode = $this->tokenError == '' ? 200 : 401;
if (!empty($this->tokenError)) {
$this->response([
'code' => 0,
'msg' => $this->tokenError
]);
}
}
// 检查token
public function checkToken()
{
$request = Request::instance();
$path = $request->path();
$tokenStr = $request->header('Authorization');
do {
//echo '所有路由都需要执行' . $path;
// 不在排除的路由列表当中都需要进行验证
if (!in_array($path, $this->notNeedToken)) {
if (empty($tokenStr)) {
// 没有传递token
$this->tokenError = '请先登录后操作';
break;
}
[$a, $token] = explode(" ", $tokenStr);
// 如果为空 直接返回错误
if (empty($token)) {
$this->tokenError = '请先登录后操作';
break;
}
// 进入验证操作
$rst = checkToken($token);
if ($rst['code'] == 2) {
$this->tokenError = $rst['msg'];
break;
}
// token本身还没有过期
$this->tokenInfo = $rst['data'];
// 检查数据库表当中是否已经标记为过期了(可能执行了退出登录了)
$appUserToken = model('api/app_user_token');
$rst = $appUserToken::get([
'user_id' => $this->tokenInfo->id,
'token' => $token
]);
// 不存在该信息
if (empty($rst)) {
$this->tokenError = '没有该登录信息';
break;
}
if ($rst['is_overtime'] == 1) {
$this->tokenError = '登录已过期';
break;
}
}
} while (0);
}
protected function response($data = [])
{
$type = $this->getResponseType();
$response = Response::create($data, $type);
throw new HttpResponseException($response);
}
}
控制器UserController
php
class User extends BaseController
{
//不需要验证token的接口
protected $notNeedToken = [
'api/user/reg',
'api/user/login'
];
}