美文网首页
使用JWT实现Token认证(PHP)

使用JWT实现Token认证(PHP)

作者: flamez57 | 来源:发表于2020-01-14 16:53 被阅读0次

为什么使用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));

相关文章

  • Spring Boot实战之Filter实现使用JWT进行接口认

    Spring Boot实战之Filter实现使用JWT进行接口认证 jwt(json web token) 用户发...

  • 使用JWT实现Token认证(PHP)

    为什么使用JWT? 随着技术的发展,分布式web应用的普及,通过session管理用户登录状态成本越来越高,因此慢...

  • 使用JWT实现Token认证

    为什么使用JWT?随着技术的发展,分布式web应用的普及,通过session管理用户登录状态成本越来越高,因此慢慢...

  • 使用JWT实现Token认证

    为什么使用JWT? 随着技术的发展,分布式web应用的普及,通过session管理用户登录状态成本越来越高,因此慢...

  • nodejs 收藏文章

    nodejs token Oauth认证 Oauth 认证-JWT 在Nodejs中使用JSON WEB Toke...

  • 带你了解JWT

    @TOC 什么是JWT 认证方式 在JWT之前,我们用过Sssion来进行认证,也使用过Token认证。那么这两种...

  • 使用python/flask实现鉴权

    上一篇文章介绍了使用JWT协议来做token认证的功能,继续装逼下去,怎样实现一个鉴权的模块? 假设token认证...

  • SpringCloud系列之网关gateway-12.实现JWT

    1.使用JWT实现获取token和校验token 首先引入依赖 使用JWT加解密逻辑 controller层 ga...

  • JWT认证原理分析

    基于session的认证 JWT认证规则 JWT(Json Web Token) JWT原理 签发算法 校验算法 ...

  • JWT使用

    JWT公司的主流Json Web Token 令牌 如何使用,取代session,还可以运用分布式认证 JWT简介...

网友评论

      本文标题:使用JWT实现Token认证(PHP)

      本文链接:https://www.haomeiwen.com/subject/jisaactx.html