Laravel jwt 多表(多用戶端)驗證隔離的實現(xiàn)
Tips: tymon/jwt-auth 作者已通過增加 prv 字段修復(fù)這一問題#1167,但是如果你是用 dingo api + jwt 的話,該問題依然存在。#
JWT 多表驗證隔離
為什么要做隔離
當(dāng)同一個 laravel 項目有多端(移動端、管理端......)都需要使用 jwt 做用戶驗證時,如果用戶表有多個(一般都會有),就需要做 token 隔離,不然會發(fā)生移動端的 token 也能請求管理端的問題,造成用戶越權(quán)。
會引發(fā)這個問題的原因是 laravel 的 jwt token 默認(rèn)只會存儲數(shù)據(jù)表的主鍵的值,并沒有區(qū)分是那個表的。所以只要 token 里攜帶的 ID 在你的用戶表中都存在,就會導(dǎo)致越權(quán)驗證。
我們來看看 laravel 的 jwt token 的原貌:
{ "iss": "http://your-request-url", "iat": 1558668215, "exp": 1645068215, "nbf": 1558668215, "jti": "XakIDuG7K0jeWGDi", "sub": 1 }
攜帶數(shù)據(jù)的是 sub 字段,其他字段是 jwt 的驗證字段。
我們只看到 sub 的值為 1,并沒有說明是那個表或是哪個驗證器的。這個 token 通過你的驗證中間件時,你使用不同的 guard 就能拿到對應(yīng)表 id 為 1 的用戶(了解 guard 請查看 laravel 的文檔)。
解決辦法
想要解決用戶越權(quán)的問題,我們只要在 token 上帶上我們的自定義字段,用來區(qū)分是哪個表或哪個驗證器生成的,然后再編寫自己的中間件驗證我們的自定義字段是否符合我們的預(yù)期。
添加自定義信息到 token
我們知道要使用 jwt 驗證,用戶模型必須要實現(xiàn) JWTSubject 的接口(代碼取自jwt 文檔):
<?php namespace App; use Tymon\JWTAuth\Contracts\JWTSubject; use Illuminate\Notifications\Notifiable; use Illuminate\Foundation\Auth\User as Authenticatable; class User extends Authenticatable implements JWTSubject { use Notifiable; // Rest omitted for brevity /** * Get the identifier that will be stored in the subject claim of the JWT. * * @return mixed */ public function getJWTIdentifier() { return $this->getKey(); } /** * Return a key value array, containing any custom claims to be added to the JWT. * * @return array */ public function getJWTCustomClaims() { return []; } }
我們可以看看實現(xiàn)的這兩個方法的作用:
- getJWTIdentifier 的:獲取會儲存到 jwt 聲明中的標(biāo)識,其實就是要我們返回標(biāo)識用戶表的主鍵字段名稱,這里是返回的是主鍵 'id',
- getJWTCustomClaims:返回包含要添加到 jwt 聲明中的自定義鍵值對數(shù)組,這里返回空數(shù)組,沒有添加任何自定義信息。
接下來我們就可以在實現(xiàn)了 getJWTCustomClaims 方法的用戶模型中添加我們的自定義信息了。
管理員模型:
/** * 額外在 JWT 載荷中增加的自定義內(nèi)容 * * @return array */ public function getJWTCustomClaims() { return ['role' => 'admin']; }
移動端用戶模型:
/** * 額外在 JWT 載荷中增加的自定義內(nèi)容 * * @return array */ public function getJWTCustomClaims() { return ['role' => 'user']; }
這里添加了一個角色名作為用戶標(biāo)識。
這樣管理員生成的 token 會像這樣:
{ "iss": "http://your-request-url", "iat": 1558668215, "exp": 1645068215, "nbf": 1558668215, "jti": "XakIDuG7K0jeWGDi", "sub": 1, "role": "admin" }
移動端用戶生成的 token 會像這樣:
{ "iss": "http://your-request-url", "iat": 1558668215, "exp": 1645068215, "nbf": 1558668215, "jti": "XakIDuG7K0jeWGDi", "sub": 1, "role": "user" }
我們可以看到這里多了一個我們自己加的 role 字段,并且對應(yīng)我們的用戶模型。
接下來我們自己寫一個中間件,解析 token 后判斷是否是我們想要的角色,對應(yīng)就通過,不對應(yīng)就報 401 就好了。
編寫 jwt 角色校驗中間件
這里提供一個可全局使用的中間件 (推薦用在用戶驗證中間件前):
<?php /** * Created by PhpStorm. * User: wlalala * Date: 2019-04-17 * Time: 13:55 */ namespace App\Http\Middleware; use Closure; use Symfony\Component\HttpKernel\Exception\UnauthorizedHttpException; use Tymon\JWTAuth\Exceptions\JWTException; use Tymon\JWTAuth\Http\Middleware\BaseMiddleware; class JWTRoleAuth extends BaseMiddleware { /** * Handle an incoming request. * * @param $request * @param Closure $next * @param null $role * @return mixed */ public function handle($request, Closure $next, $role = null) { try { // 解析token角色 $token_role = $this->auth->parseToken()->getClaim('role'); } catch (JWTException $e) { /** * token解析失敗,說明請求中沒有可用的token。 * 為了可以全局使用(不需要token的請求也可通過),這里讓請求繼續(xù)。 * 因為這個中間件的責(zé)職只是校驗token里的角色。 */ return $next($request); } // 判斷token角色。 if ($token_role != $role) { throw new UnauthorizedHttpException('jwt-auth', 'User role error'); } return $next($request); } }
注冊 jwt 角色校驗中間件
在 app/Http/Kernel.php 中注冊中間件:
/** * The application's route middleware. * * These middleware may be assigned to groups or used individually. * * @var array */ protected $routeMiddleware = [ // ...省略 ... // 多表jwt驗證校驗 'jwt.role' => \App\Http\Middleware\JWTRoleAuth::class, ];
使用 jwt 角色校驗中間件
接下來在需要用戶驗證的路由組中添加我們的中間件:
Route::group([ 'middleware' => ['jwt.role:admin', 'jwt.auth'], ], function ($router) { // 管理員驗證路由 // ... }); Route::group([ 'middleware' => ['jwt.role:user', 'jwt.auth'], ], function ($router) { // 移動端用戶驗證路由 // ... });
至此完成 jwt 多表用戶驗證隔離。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:PHP利用緩存處理用戶注冊時的郵箱驗證,成功后用戶數(shù)據(jù)存入數(shù)據(jù)庫操作示例
欄 目:PHP編程
本文標(biāo)題:Laravel jwt 多表(多用戶端)驗證隔離的實現(xiàn)
本文地址:http://mengdiqiu.com.cn/a1/PHPbiancheng/11010.html
您可能感興趣的文章
- 01-11Laravel 微信小程序后端搭建步驟詳解
- 01-11Laravel框架Blade模板簡介及模板繼承用法分析
- 01-11Laravel 微信小程序后端實現(xiàn)用戶登錄的示例代碼
- 01-11Laravel框架基礎(chǔ)語法與知識點整理【模板變量、輸出、include引入
- 01-11Laravel框架Eloquent ORM刪除數(shù)據(jù)操作示例
- 01-11Laravel框架Eloquent ORM修改數(shù)據(jù)操作示例
- 01-11Laravel框架Eloquent ORM簡介、模型建立及查詢數(shù)據(jù)操作詳解
- 01-11laravel5.1框架下的批量賦值實現(xiàn)方法分析
- 01-11laravel5.5框架的上傳圖片功能實例分析【僅傳到服務(wù)器端】
- 01-11Laravel框架下載,安裝及路由操作圖文詳解


閱讀排行
本欄相關(guān)
- 04-02php本站才可以請求數(shù)據(jù) php本地數(shù)據(jù)庫
- 04-02關(guān)于txt數(shù)據(jù)庫php的信息
- 04-02php打印請求數(shù)據(jù) php打印輸出結(jié)果
- 04-02網(wǎng)頁里php操作數(shù)據(jù)庫 php網(wǎng)頁例子
- 04-02php插入數(shù)據(jù)庫為亂碼 php連接數(shù)據(jù)庫亂
- 04-02php數(shù)據(jù)庫地址 phpstudy 數(shù)據(jù)庫
- 04-02php數(shù)據(jù)庫數(shù)據(jù)相加 php數(shù)據(jù)庫添加數(shù)據(jù)
- 04-02數(shù)據(jù)權(quán)限架構(gòu)思路php 數(shù)據(jù)權(quán)限設(shè)計方
- 04-02php數(shù)據(jù)庫輸入變量 php里輸出數(shù)據(jù)庫數(shù)
- 04-02php如何用導(dǎo)入數(shù)據(jù) php用來導(dǎo)入其他文
隨機閱讀
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-10delphi制作wav文件的方法
- 04-02jquery與jsp,用jquery
- 01-11ajax實現(xiàn)頁面的局部加載
- 01-10C#中split用法實例總結(jié)
- 01-10使用C語言求解撲克牌的順子及n個骰子