zoukankan      html  css  js  c++  java
  • jwt鉴权学习 (php示例代码)

      前段时间听朋友讲起 jwt鉴权,博主我是一脸懵逼,通过朋友坚持不懈的讲解,我终于听懂了,jwt就是登陆token校验嘛

      然而事情并不是博主想象的那么简单,在一个艳阳高照,晴空万里的夜晚,博主手贱百度了一波jwt,才发现问题所在之处

      首先:我的登录token校验逻辑

          用户输入账号密码登录,账号密码校验正确,生成一个随机拼接的token,返回给前端,并存储到数据库中,然后根据前端发送的token和数据库的token对比,从而达到校验效果

          (缺点,token永不过期,如果设置了定时清空所有token,用户体验极差)

      百度上大佬们的博客讲解jwt鉴权的校验逻辑

          用户输入账号密码登录,账号密码校验正确,使用n个数据(详情看后面的代码)组成的数组,进行加密生成字符串,拼接上加密算法名称类型加密的字符串,自己设置

        的秘钥加前面两个数据再进行加密拼接,于是乎就组成了一个token,并返回,服务端并不存储任何数据   

          生成token的示例 :  aaa.bbb.ccc        (加密后的数据a+ . +加密后的数据b+ . +(加密(加密前的数据a+加密前的数据b+秘钥)))

          当用户用token发起请求的时候,服务端则会拿到token,解密再进行校验,token的有效时间是否过期什么的

     下面附上代码:

       

    <?php
    /**
     * PHP实现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',  //请求生成token的用户
         *  'iat'=>time(),  //生成时间
         *  'exp'=>time()+7200,  //过期时间
         *  'nbf'=>time()+60,  //生成token  60秒后方可使用
         *  'sub'=>'www.admin.com',  //服务端网址
         *  'jti'=>md5(uniqid('JWT').time())  //该Token唯一标识
         * ]
         * @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是否有效,默认验证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);
    
            //签发时间大于当前服务器时间验证失败
            if (isset($payload['iat']) && $payload['iat'] > time())
                return false;
    
            //过期时间小宇当前服务器时间验证失败
            if (isset($payload['exp']) && $payload['exp'] < time())
                return false;
    
            //该nbf时间之前不接收处理该Token
            if (isset($payload['nbf']) && $payload['nbf'] > time())
                return false;
    
            return $payload;
        }
    
    
    
    
        /**
         * base64UrlEncode   https://jwt.io/  中base64UrlEncode编码实现
         * @param string $input 需要编码的字符串
         * @return string
         */
        private static function base64UrlEncode($input)
        {
            return str_replace('=', '', strtr(base64_encode($input), '+/', '-_'));
        }
    
        /**
         * base64UrlEncode  https://jwt.io/  中base64UrlEncode解码实现
         * @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签名实现
         * @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));
        }
    }
    
    
    
    
    
        //例子一
        //生成token
        $payload=array('sub'=>'1234567890','name'=>'John Doe','iat'=>1516239022);
        $jwt=new Jwt;
        $token=$jwt->getToken($payload);
        echo "<pre>";
        echo $token;
        //生成的token
        // eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.keH6T3x1z7mmhKL1T3r9sQdAxxdzB6siemGMr_6ZOwU
    
        //校验token
        $getPayload=$jwt->verifyToken($token);
        echo "<br><br>";
        var_dump($getPayload);
        echo "<br><br>";
        //解密后的返回值
        //    array(3) {
        //        ["sub"]=>
        //      string(10) "1234567890"
        //        ["name"]=>
        //      string(8) "John Doe"
        //        ["iat"]=>
        //      int(1516239022)
        //    }
    
    
    
    
    
    
        //例子二
        //生成token
        $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值
        //  eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJhZG1pbiIsImlhdCI6MTU4Mzk4MTc2NywiZXhwIjoxNTgzOTg4OTY3LCJuYmYiOjE1ODM5ODE3NjcsInN1YiI6Ind3dy5hZG1pbi5jb20iLCJqdGkiOiJhMzdiOTRlZTc5YzE2ZDI2YTA1NWY3OGI2Mzg0M2YwOSJ9.RyHdg5lTsnquY0G_q053DtR5FnSbBrU_GdY_j3GPLcs
    
    
        //校验token
        $getPayload_test=Jwt::verifyToken($token_test);
        echo "<br><br>";
        var_dump($getPayload_test);
        echo "<br><br>";
        //解密后的返回值
        //    array(6) {
        //        ["iss"]=>
        //      string(5) "admin"
        //        ["iat"]=>
        //      int(1583981767)
        //      ["exp"]=>
        //      int(1583988967)
        //      ["nbf"]=>
        //      int(1583981767)
        //      ["sub"]=>
        //      string(13) "www.admin.com"
        //        ["jti"]=>
        //      string(32) "a37b94ee79c16d26a055f78b63843f09"
        //    }

    原文地址:https://blog.csdn.net/qq_33858250/article/details/89419869

  • 相关阅读:
    vim初试(Hello World)
    CSP201809-2 买菜(超简单的方法!!)
    CSP202006-2 稀疏向量
    CSP202012-2 期末预测之最佳阈值
    浮点数表示
    结构体
    全排列-康托展开及逆展开
    CA-031 上手Games101环境 Games101环境怎么配置
    计算机图形学 实验四 AET算法
    计算机图形学 实验三 梁氏裁剪算法
  • 原文地址:https://www.cnblogs.com/junyi-bk/p/12468272.html
Copyright © 2011-2022 走看看