使用JWT实现Token认证(PHP)

2020-01-14  本文已影响0人  flamez57

为什么使用JWT?

随着技术的发展,分布式web应用的普及,通过session管理用户登录状态成本越来越高,因此慢慢发展成为token的方式做登录身份校验,然后通过token去取redis中的缓存的用户信息,随着之后jwt的出现,校验方式更加简单便捷化,无需通过redis缓存,而是直接根据token取出保存的用户信息,以及对token可用性校验,单点登录更为简单。

JWT架构图

JWT架构图

https://github.com/cdoco/php-jwt

php实现代码

<?php

class Jwt

{

    /**

    ** 头部

    ** @var  $header array

    */

    protected static $header = [

        'alg' => 'HS256', //生成signature的算法

        'typ' => 'JWT'  //类型

    ];

    protected static $key;

    protected static $time;

    public function __construct()

    {

        self::$key = 'secret'; //密钥

        self::$time = time();

    }

    /**

    ** 获取jwt token

    ** @param array $payload jwt载荷  格式如下非必须

    ** @return bool|string 返回结果集

    **/

    public static function getToken(array $payload)

    {

        $arr = [

            'iss' => 'mysweet95', //该JWT的签发者

            'iat' => self::$time, //签发时间

            'exp' => self::$time + 15, //过期时间

            'nbf' => self::$time, //该时间之前不接收处理该Token

            'sub' => '', //面向的用户

            'jti' => md5(uniqid('JWT') . self::$time) //该Token唯一标识

        ];

        $payload = array_merge($arr, $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是否有效,默认验证exp,nbf,iat时间

    ** @param string $Token 需要验证的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;

        }

        //签名验证

        if (self::signature($base64header . '.' . $base64payload, self::$key, $base64decodeheader['alg']) !== $sign) {

            return false;

        }

        $payload = json_decode(self::base64UrlDecode($base64payload), JSON_OBJECT_AS_ARRAY);

        //签发时间大于当前服务器时间验证失败

        if (isset($payload['iat']) && $payload['iat'] > self::$time) {

            return false;

        }

        //过期时间小宇当前服务器时间验证失败

        if (isset($payload['exp']) && $payload['exp'] < self::$time) {

            return false;

        }

        //该nbf时间之前不接收处理该Token

        if (isset($payload['nbf']) && $payload['nbf'] > self::$time) {

            return false;

        }

        return $payload;

    }

    /**

    ** base64UrlEncode  https://jwt.io/ 中base64UrlEncode编码实现

    ** @param string $input 需要编码的字符串

    ** @return mixed 返回结果集

    */

    private static function base64UrlEncode(string $input)

    {

        return str_replace('=', '', strtr(base64_encode($input), '+/', '-_'));

    }

    /**

    **  base64UrlEncode https://jwt.io/ 中base64UrlEncode解码实现

    ** @param string $input 需要解码的字符串

    ** @return false|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签名实现

    ** @param string $input 为base64UrlEncode(header).".".base64UrlEncode(payload)

    ** @param string $key

    ** @param string $alg

    ** @return mixed 返回结果集

    */

    private static function signature(string $input, string $key, $alg = "HS256")

    {

        $alg_config = ['HS256' => 'sha256'];

        return self::base64UrlEncode(hash_hmac($alg_config[$alg], $input, $key, true));

    }

}

//===测试使用

new Jwt();

var_dump(Jwt::getToken(['a' => 'sdfs']));

$a = 'eyJhbGciOiYmYiOjEJf8xRlnXZFCIwDFJfFSJd57bWph-zuM';

var_dump(Jwt::verifyToken($a));

上一篇 下一篇

猜你喜欢

热点阅读