php 模数 指数 公钥生成_php实现JWT认证
什么是JWTJWT(json web token)是為了在網(wǎng)絡(luò)應(yīng)用環(huán)境間傳遞聲明而執(zhí)行的一種基于JSON的開放標(biāo)準(zhǔn)。
JWT的聲明一般被用來在身份提供者和服務(wù)提供者間傳遞被認(rèn)證的用戶身份信息,以便于從資源服務(wù)器獲取資源。比如用在用戶登錄上。
JWT定義了一種用于簡潔,自包含的用于通信雙方之間以 JSON 對(duì)象的形式安全傳遞信息的方法。JWT 可以使用 HMAC 算法或者是 RSA 的公鑰密鑰對(duì)進(jìn)行簽名。
JWT有兩個(gè)特點(diǎn):
自包含(Self-contained):負(fù)載中包含了所有用戶所需要的信息,避免了多次查詢數(shù)據(jù)庫
簡潔(Compact):可以通過URL, POST 參數(shù)或者在 HTTP header 發(fā)送,因?yàn)閿?shù)據(jù)量小,傳輸速度快
JWT組成
JWT由header,payload,signature三個(gè)部分,下面我們用官網(wǎng)的實(shí)例先來講解一個(gè)這三個(gè)部分的用法。
header部分:
jwt的頭部承載兩部分信息:
聲明類型,這里是jwt
聲明加密的算法 通常直接使用 HMAC SHA256
完整的頭部就像下面這樣的JSON:
{"alg": "HS256","typ": "JWT" }對(duì)應(yīng)base64UrlEncode編碼為:
eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9說明:該字段為json格式。alg字段指定了生成signature的算法,默認(rèn)值為HS256,typ默認(rèn)值為JWT
payload部分:
載荷就是存放有效信息的地方。
標(biāo)準(zhǔn)中注冊(cè)的聲明 (建議但不強(qiáng)制使用) :
iss: jwt簽發(fā)者
sub: jwt所面向的用戶
aud: 接收jwt的一方
exp: jwt的過期時(shí)間,這個(gè)過期時(shí)間必須要大于簽發(fā)時(shí)間
nbf: 定義在什么時(shí)間之前,該jwt都是不可用的
iat: jwt的簽發(fā)時(shí)間
jti: jwt的唯一身份標(biāo)識(shí),主要用來作為一次性token,從而回避重放攻擊。
{"sub": "1234567890","name": "John Doe","iat": 1516239022 }對(duì)應(yīng)base64UrlEncode編碼為:
eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ說明:該字段為json格式,表明用戶身份的數(shù)據(jù),可以自己自定義字段,很靈活。sub 面向的用戶,name 姓名 ,iat 簽發(fā)時(shí)間。例如可自定義示例如下:
{"iss": "admin", //該JWT的簽發(fā)者"iat": 1535967430, //簽發(fā)時(shí)間"exp": 1535974630, //過期時(shí)間"nbf": 1535967430, //該時(shí)間之前不接收處理該Token"sub": "www.admin.com", //面向的用戶"jti": "9f10e796726e332cec401c569969e13e" //該Token唯一標(biāo)識(shí) }signature部分:
jwt的第三部分是一個(gè)簽證信息,這個(gè)簽證信息由三部分組成:
- header (base64后的)
- payload (base64后的)
- secret
對(duì)應(yīng)的簽名為:
keH6T3x1z7mmhKL1T3r9sQdAxxdzB6siemGMr_6ZOwU最終得到的JWT的json為(header.payload.signature):
eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.keH6T3x1z7mmhKL1T3r9sQdAxxdzB6siemGMr_6ZOwU說明:對(duì)header和payload進(jìn)行base64UrlEncode編碼后進(jìn)行拼接。通過key(這里是123456)進(jìn)行HS256算法簽名。
JWT的使用流程
- 初次登錄:用戶初次登錄,輸入用戶名密碼
- 密碼驗(yàn)證:服務(wù)器從數(shù)據(jù)庫取出用戶名和密碼進(jìn)行驗(yàn)證
- 生成JWT:服務(wù)器端驗(yàn)證通過,根據(jù)從數(shù)據(jù)庫返回的信息,以及預(yù)設(shè)規(guī)則,生成JWT
- 返還JWT:服務(wù)器的HTTP RESPONSE中將JWT返還
- 帶JWT的請(qǐng)求:以后客戶端發(fā)起請(qǐng)求,HTTP REQUEST
- HEADER中的Authorizatio字段都要有值,為JWT
- 服務(wù)器驗(yàn)證JWT
PHP如何實(shí)現(xiàn)JWT
這里使用的是PHP 7.0.31,我們新建一個(gè)文件jwtAuth.php,完整類代碼如下:
<?php /*** PHP實(shí)現(xiàn)jwt*/ class JwtAuth {//頭部private static $header=array('alg'=>'HS256', //生成signature的算法'typ'=>'JWT' //類型);//使用HMAC生成信息摘要時(shí)所使用的密鑰private static $key='root123456';/*** 獲取jwt token* @param array $payload jwt載荷 格式如下非必須* [* 'iss'=>'jwt_admin', //該JWT的簽發(fā)者* 'iat'=>time(), //簽發(fā)時(shí)間* 'exp'=>time()+7200, //過期時(shí)間* 'nbf'=>time()+60, //該時(shí)間之前不接收處理該Token* 'sub'=>'www.mano100.cn', //面向的用戶* 'jti'=>md5(uniqid('JWT').time()) //該Token唯一標(biāo)識(shí)* ]* @return bool|string*/public static function getToken(array $payload){if(is_array($payload)){$base64header=self::base64UrlEncode(json_encode(self::$header,JSON_UNESCAPED_UNICODE));$base64payload=self::base64UrlEncode(json_encode($payload,JSON_UNESCAPED_UNICODE));$token=$base64header.'.'.$base64payload.'.'.self::signature($base64header.'.'.$base64payload,self::$key,self::$header['alg']);return $token;}else{return false;}}/*** 驗(yàn)證token是否有效,默認(rèn)驗(yàn)證exp,nbf,iat時(shí)間* @param string $Token 需要驗(yàn)證的token* @return bool|string*/public static function verifyToken(string $Token){$tokens = explode('.', $Token);if (count($tokens) != 3)return false;list($base64header, $base64payload, $sign) = $tokens;//獲取jwt算法$base64decodeheader = json_decode(self::base64UrlDecode($base64header), JSON_OBJECT_AS_ARRAY);if (empty($base64decodeheader['alg']))return false;//簽名驗(yàn)證if (self::signature($base64header . '.' . $base64payload, self::$key, $base64decodeheader['alg']) !== $sign)return false;$payload = json_decode(self::base64UrlDecode($base64payload), JSON_OBJECT_AS_ARRAY);//簽發(fā)時(shí)間大于當(dāng)前服務(wù)器時(shí)間驗(yàn)證失敗if (isset($payload['iat']) && $payload['iat'] > time())return false;//過期時(shí)間小宇當(dāng)前服務(wù)器時(shí)間驗(yàn)證失敗if (isset($payload['exp']) && $payload['exp'] < time())return false;//該nbf時(shí)間之前不接收處理該Tokenif (isset($payload['nbf']) && $payload['nbf'] > time())return false;return $payload;}/*** base64UrlEncode https://jwt.io/ 中base64UrlEncode編碼實(shí)現(xiàn)* @param string $input 需要編碼的字符串* @return string*/private static function base64UrlEncode(string $input){return str_replace('=', '', strtr(base64_encode($input), '+/', '-_'));}/*** base64UrlEncode https://jwt.io/ 中base64UrlEncode解碼實(shí)現(xiàn)* @param string $input 需要解碼的字符串* @return bool|string*/private static function base64UrlDecode(string $input){$remainder = strlen($input) % 4;if ($remainder) {$addlen = 4 - $remainder;$input .= str_repeat('=', $addlen);}return base64_decode(strtr($input, '-_', '+/'));}/*** HMACSHA256簽名 https://jwt.io/ 中HMACSHA256簽名實(shí)現(xiàn)* @param string $input 為base64UrlEncode(header).".".base64UrlEncode(payload)* @param string $key* @param string $alg 算法方式* @return mixed*/private static function signature(string $input, string $key, string $alg = 'HS256'){$alg_config=array('HS256'=>'sha256');return self::base64UrlEncode(hash_hmac($alg_config[$alg], $input, $key,true));} }這里測試一下
//測試和官網(wǎng)是否匹配begin$payload=array('sub'=>'1234567890','name'=>'John Doe','iat'=>1516239022);$jwt=new Jwt;$token=$jwt->getToken($payload);echo "<pre>";echo $token;//對(duì)token進(jìn)行驗(yàn)證簽名$getPayload=$jwt->verifyToken($token);echo "<br><br>";var_dump($getPayload);echo "<br><br>";//測試和官網(wǎng)是否匹配end//自己使用測試begin$payload_test=array('iss'=>'admin','iat'=>time(),'exp'=>time()+7200,'nbf'=>time(),'sub'=>'www.admin.com','jti'=>md5(uniqid('JWT').time()));;$token_test=Jwt::getToken($payload_test);echo "<pre>";echo $token_test;//對(duì)token進(jìn)行驗(yàn)證簽名$getPayload_test=Jwt::verifyToken($token_test);echo "<br><br>";var_dump($getPayload_test);echo "<br><br>";//自己使用時(shí)候end以上內(nèi)容希望幫助到大家,很多PHPer在進(jìn)階的時(shí)候總會(huì)遇到一些問題和瓶頸,業(yè)務(wù)代碼寫多了沒有方向感,不知道該從那里入手去提升,對(duì)此我整理了一些資料,包括但不限于:分布式架構(gòu)、高可擴(kuò)展、高性能、高并發(fā)、服務(wù)器性能調(diào)優(yōu)、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql優(yōu)化、shell腳本、Docker、微服務(wù)、Nginx等多個(gè)知識(shí)點(diǎn)高級(jí)進(jìn)階干貨需要的可以免費(fèi)分享給大家,需要
PHP進(jìn)階架構(gòu)師>>>視頻、面試文檔免費(fèi)獲取?docs.qq.com或者關(guān)注咱們下面的專欄
PHP架構(gòu)師之路?zhuanlan.zhihu.com來源:https://zhuanlan.zhihu.com/p/93910087總結(jié)
以上是生活随笔為你收集整理的php 模数 指数 公钥生成_php实现JWT认证的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 乐高ev3搭建图_乐高EV3机械爪合集
- 下一篇: 地磅称重软件源码_电脑设备器件+塔吊主吊