生活随笔
收集整理的這篇文章主要介紹了
php中使用JWT
小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
https://baijiahao.baidu.com/s?id=1608021814182894637&wfr=spider&for=pc
https://www.jb51.net/article/146790.htm
JWT是什么
JWT是json web token縮寫。它將用戶信息加密到token里,服務(wù)器不保存任何用戶信息。服務(wù)器通過使用保存的密鑰驗證token的正確性,只要正確即通過驗證。基于token的身份驗證可以替代傳統(tǒng)的cookie+session身份驗證方法。
它定義了一種用于簡潔,自包含的用于通信雙方之間以 JSON 對象的形式安全傳遞信息的方法。JWT 可以使用 HMAC 算法或者是 RSA 的公鑰密鑰對進(jìn)行簽名。它具備兩個特點:
簡潔(Compact):可以通過URL, POST 參數(shù)或者在 HTTP header 發(fā)送,因為數(shù)據(jù)量小,傳輸速度快
自包含(Self-contained):負(fù)載中包含了所有用戶所需要的信息,避免了多次查詢數(shù)據(jù)庫
JWT由三個部分組成:header.payload.signature
以下示例以JWT官網(wǎng)為例
header部分:
{"alg": "HS256","typ": "JWT"
}
對應(yīng)base64UrlEncode編碼為:eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9
說明:該字段為json格式。alg字段指定了生成signature的算法,默認(rèn)值為HS256,typ默認(rèn)值為JWT
payload部分:
{"sub": "1234567890","name": "John Doe","iat": 1516239022
}
對應(yīng)base64UrlEncode編碼為:eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ
說明:該字段為json格式,表明用戶身份的數(shù)據(jù),可以自己自定義字段,很靈活。sub 面向的用戶,name 姓名 ,iat 簽發(fā)時間。例如可自定義示例如下:
{"iss": "admin", //該JWT的簽發(fā)者"iat": 1535967430, //簽發(fā)時間"exp": 1535974630, //過期時間"nbf": 1535967430, //該時間之前不接收處理該Token"sub": "www.admin.com", //面向的用戶"jti": "9f10e796726e332cec401c569969e13e" //該Token唯一標(biāo)識
}
signature部分:
HMACSHA256(base64UrlEncode(header) + "." +base64UrlEncode(payload),123456
)
對應(yīng)的簽名為:keH6T3x1z7mmhKL1T3r9sQdAxxdzB6siemGMr_6ZOwU
最終得到的JWT的json為(header.payload.signature):eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.keH6T3x1z7mmhKL1T3r9sQdAxxdzB6siemGMr_6ZOwU
說明:對header和payload進(jìn)行base64UrlEncode編碼后進(jìn)行拼接。通過key(這里是123456)進(jìn)行HS256算法簽名。
PHP如何實現(xiàn)JWT
作者使用的是PHP 7.0.31,不廢話,直接上代碼,新建jwt.php,復(fù)制粘貼如下
<?php
/*** PHP實現(xiàn)jwt*/
class Jwt {//頭部private static $header=array('alg'=>'HS256', //生成signature的算法'typ'=>'JWT' //類型);//使用HMAC生成信息摘要時所使用的密鑰private static $key='123456';/*** 獲取jwt token* @param array $payload jwt載荷 格式如下非必須* [* 'iss'=>'jwt_admin', //該JWT的簽發(fā)者* 'iat'=>time(), //簽發(fā)時間* 'exp'=>time()+7200, //過期時間* 'nbf'=>time()+60, //該時間之前不接收處理該Token* 'sub'=>'www.admin.com', //面向的用戶* 'jti'=>md5(uniqid('JWT').time()) //該Token唯一標(biāo)識* ]* @return bool|string*/public static function getToken( $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;}}/*** 驗證token是否有效,默認(rèn)驗證exp,nbf,iat時間* @param string $Token 需要驗證的token* @return bool|string*/public static function verifyToken( $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;//簽名驗證if (self::signature($base64header . '.' . $base64payload, self::$key, $base64decodeheader['alg']) !== $sign)return false;$payload = json_decode(self::base64UrlDecode($base64payload), JSON_OBJECT_AS_ARRAY);//簽發(fā)時間大于當(dāng)前服務(wù)器時間驗證失敗if (isset($payload['iat']) && $payload['iat'] > time())return false;//過期時間小宇當(dāng)前服務(wù)器時間驗證失敗if (isset($payload['exp']) && $payload['exp'] < time())return false;//該nbf時間之前不接收處理該Tokenif (isset($payload['nbf']) && $payload['nbf'] > time())return false;return $payload;}/*** base64UrlEncode https://jwt.io/ 中base64UrlEncode編碼實現(xiàn)* @param string $input 需要編碼的字符串* @return string*/private static function base64UrlEncode( $input){return str_replace('=', '', strtr(base64_encode($input), '+/', '-_'));}/*** base64UrlEncode https://jwt.io/ 中base64UrlEncode解碼實現(xiàn)* @param string $input 需要解碼的字符串* @return bool|string*/private static function base64UrlDecode( $input){$remainder = strlen($input) % 4;if ($remainder) {$addlen = 4 - $remainder;$input .= str_repeat('=', $addlen);}return base64_decode(strtr($input, '-_', '+/'));}/*** HMACSHA256簽名 https://jwt.io/ 中HMACSHA256簽名實現(xiàn)* @param string $input 為base64UrlEncode(header).".".base64UrlEncode(payload)* @param string $key* @param string $alg 算法方式* @return mixed*/private static function signature( $input, $key, $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;//對token進(jì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;//對token進(jìn)行驗證簽名
$getPayload_test=Jwt::verifyToken($token_test);
echo "<br><br>";
var_dump($getPayload_test);
echo "<br><br>";
//自己使用時候end
總結(jié)
以上是生活随笔為你收集整理的php中使用JWT的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。