thinkphp6 JWT报错 ‘“kid“ empty, unable to lookup correct key‘解决办法

文章目录

    • JWT简介
    • 安装
    • 问题
    • 先前的代码
    • 解决办法
    • 修改后的完整代码

JWT简介

JWT全称为Json Web
Token,是一种用于在网络应用之间传递信息的简洁、安全的方式。JWT标准定义了一种简洁的、自包含的方法用于通信双方之间以JSON对象的形式安全的传递信息。由于它的简洁性、可扩展性和可传递性,成为当前最流行的跨域身份验证解决方案之一。一般来说,JWT会被用来在客户端和服务端之间传递一些敏感信息,比如用户认证信息、权限信息等等。

安装

thinkphp框架默认不带这个依赖包,需要自己安装

composer require firebase/php-jwt

使用的时候直接use即可

use \Firebase\JWT\JWT;

问题

thinkphp6使用JWT报错 ‘“kid“ empty, unable to lookup correct key‘
这个玩意简直就是一个大坑,弄了一个早上
下面是报错的内容
在这里插入图片描述
遇到这个问题我也很无语
最后发现是JWT包的encode 和 decode 和之前的有区别

先前的代码

 public function index()
    {
        $username = Request::post('username');
        $password = Request::post('password');
        $user = User::where('username', $username)->find();
        if (!$user) {
            return json([
                'code' => 401,
                'message' => '用户名错误,请重试',
                'data' => null
            ]);
        }
        if ($user->password !== md5($password)) {
            return json([
                'code' => 401,
                'message' => '密码错误,请重试',
                'data' => null
            ]);
        }
        // 生成 token
        $key = '147258369'; // 生成 token 的秘钥
        $payload = array(
            // "iss" => "http://127.0.0.1:8000",  // JWT的签发者
            // "aud" => "http://127.0.0.1:9528/",  // JWT的接收者可以省略
            "iat" => time(),  // token 的创建时间
            "nbf" =>  time(),  // token 的生效时间
            "exp" => time() + 3600,  // token 的过期时间
            "data" => [
                // 包含的用户信息等数据
                "username" => $username,
                "permission"=>$user->permission
            ]
        );
        $jwt = JWT::encode($payload, $key,'HS256');
        return json([
            'code' => 200,
            'message' => '登录成功',
            'token' => $jwt,
            'data' => [
                'id' => $user->id,
                'username' => $user->username,
                'permission' => $user->permission,
            ]
        ]);
    }

这是获取用户信息的方法
同时包含了对JWT字符串进行解析的操作

 public function getInfo(){
        $key = '147258369';
        $token = Request::get('token'); // 需要验证的 Token 字符串

            $decoded = JWT::decode($token,$key,['HS256']);// 解码并验证 Token
            if(!$decoded) return json([
                
                    'code' => 401,
                    'message' => 'token无效',
                    'data' => null
            ]);
            // 如果 Token 有效,$decoded 会包含解码后的数据
            $username = $decoded->data->username;
            $permission = $decoded->data->permission;
            // 验证通过,执行相应操作
                return json([
                'code' => 200,
                'message' => '获取用户信息成功',
                'data' => [
                    'username' => $username,
                    'permission' => $permission,
                ]
            ]);
        }

乍一看好像没什么问题,是的换做以前确实没有问题,但是现在这种写法好像用不了

解决办法

只需要在生成JWT的时候添加一个keyId
具体内容如下

$keyId = "keyId";
 $jwt = JWT::encode($payload, $key,'HS256',$keyId);

在解析的时候key要用new Key的方法

use Firebase\JWT\Key;

$key = new Key('key', 'HS256');
$decode = JWT::decode($token, $key);

修改后的完整代码

重点就是生成JWT的时候要记得加上keyId
在解析的时候key要用new Key的方法

生成JWT

<?php
namespace app\controller;

use app\BaseController;
use think\facade\Request;
use app\model\User;
use \Firebase\JWT\JWT;

class Login extends BaseController
{
    public function index()
    {
        $username = Request::post('username');
        $password = Request::post('password');
        $user = User::where('username', $username)->find();
        if (!$user) {
            return json([
                'code' => 401,
                'message' => '用户名错误,请重试',
                'data' => null
            ]);
        }
        if ($user->password !== md5($password)) {
            return json([
                'code' => 401,
                'message' => '密码错误,请重试',
                'data' => null
            ]);
        }
        // 生成 token
        $key = '147258369'; // 生成 token 的秘钥
        $payload = array(
            // "iss" => "http://127.0.0.1:8000",  // issuer
            // "aud" => "http://example.com",  // audience
            "iat" => time(),  // token 的创建时间
            "nbf" =>  time(),  // token 的生效时间
            "exp" => time() + 3600,  // token 的过期时间
            "data" => [
                // 包含的用户信息等数据
                "username" => $username,
                "permission"=>$user->permission
            ]
        );
        $keyId = "keyId"; //这个东西必须要加上,不加上,报错,报错内容:'"kid" empty, unable to lookup correct key'
        $jwt = JWT::encode($payload, $key,'HS256',$keyId);
        return json([
            'code' => 200,
            'message' => '登录成功',
            'token' => $jwt,
            'data' => [
                'id' => $user->id,
                'username' => $user->username,
                'permission' => $user->permission,
            ]
        ]);
    }
}

解析JWT

<?php
namespace app\controller;
use app\BaseController;
use think\facade\Request;
use Firebase\JWT\JWT;
use Firebase\JWT\Key;

class User extends BaseController{
    public function getInfo(){
        $key = new Key('147258369', 'HS256');   // 密钥
        // $key = '147258369';
        $token = Request::get('token'); // 需要验证的 Token 字符串

            $decoded = JWT::decode($token,$key);// 解码并验证 Token
            if(!$decoded) return json([
                
                    'code' => 401,
                    'message' => 'token无效',
                    'data' => null
            ]);
            // 如果 Token 有效,$decoded 会包含解码后的数据
            $username = $decoded->data->username;
            $permission = $decoded->data->permission;
            // 验证通过,执行相应操作
                return json([
                'code' => 200,
                'message' => '获取用户信息成功',
                'data' => [
                    'username' => $username,
                    'permission' => $permission,
                ]
            ]);
        }
}

完结!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/16954.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

关于SpringBoot整合Websocket实现简易对话聊天窗

前言 官网链接&#xff1a;Websocket Websocket 是什么&#xff1f;它可以将两个独立的浏览器窗口作为通信的两端。 这种形式的通信与传统的 HTTP、TCP 所不同。传统的 HTTP 请求—响应协议是无法实现实时通信的&#xff0c;也就是说&#xff0c;只能由客户端向服务端发送请求…

英语中主语从句的概念及其用法,例句(不断更新)

主语从句的原理 主语从句是一种充当整个句子主语的从句&#xff0c;主语从句构成的句子&#xff0c;是要以引导词开头的。它可以用名词性从属连词、关系代词或关系副词引导。主语从句通常位于谓语动词之前&#xff0c;用于表示动作、状态或事件的主体。 以下是一些常用的引导主…

MiniGPT-4,开源了!

上个月GPT-4发布时&#xff0c;我曾写过一篇文章分享过有关GPT-4的几个关键信息。 当时的分享就提到了GPT-4的一个重要特性&#xff0c;那就是多模态能力。 比如发布会上演示的&#xff0c;输入一幅图&#xff08;手套掉下去会怎么样&#xff1f;&#xff09;。 GPT-4可以理解…

推荐几个可以免费使用的ChatGPT工具

在ChatGPT相关API推出之后&#xff0c;各种工具如雨后春笋一般层出不穷&#xff0c;这篇文章就列举一些日常使用到的工具。 工具列表 OpenAI 在线读取任意网页内容包括视频&#xff08;YouTube&#xff09;&#xff0c;并根据这些内容回答你提出的相关问题或总结相关内容支持…

Mysql-视图

视图 视图介绍视图的语法视图的检查选项CASCADEDLOCAL 视图的更新视图的作用 视图介绍 视图&#xff08;View&#xff09;是一种虚拟存在的表。视图中的数据并不在数据库中实际存在&#xff0c;行和列数据来自定义视图的查询中使用的表&#xff0c;并且是在使用视图时动态生成的…

【配电网优化】基于串行和并行ADMM算法的配电网优化研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

2023年值得关注的20大网络安全趋势

随着围绕所有企业的数字革命&#xff0c;无论大小&#xff0c;企业、组织甚至政府都依赖计算机化系统来管理他们的日常活动&#xff0c;从而使网络安全成为保护数据免受各种在线攻击或任何未经授权访问的主要目标。 随着数据泄露、勒索软件和黑客攻击的新闻成为常态&#xff0…

java获取文件夹下所有文件名

在进行 Java编程的过程中&#xff0c;我们会经常使用到文件夹下的所有文件名。有时候可能不太熟悉 Java编程的小伙伴们会发现&#xff0c;在代码中没有获取到所有的文件名&#xff0c;那么这个时候我们应该怎么去获取到这些文件呢&#xff1f;在进行 Java编程的过程中&#xff…

深度学习卷积神经网络学习小结

————————————————————————————————————————————— 学习小结&#xff1a; 1&#xff09;深度学习综述&#xff1b;&#xff08;2&#xff09;对卷积神经网络&#xff08;CNN&#xff09;的认识&#xff1b;&#xff08;3&#xff0…

08 Kubernetes应用配置管理

课件 在 Kubernetes 中&#xff0c;secret 是一种用于存储敏感信息的对象。Kubernetes 支持以下三种类型的 secret&#xff1a; Opaque&#xff1a;这是默认的 secret 类型&#xff0c;可以用于存储任何类型的数据&#xff0c;包括字符串、二进制数据等。 Service Account&…

Python研究生组蓝桥杯(省二)参赛感受

为什么参加蓝桥杯&#xff1f; 今年是读研的第一年&#xff0c;看着我简历上的获奖经历“优秀学生干部”“优秀志愿者”“优秀毕业生”......大学四年&#xff0c;我竟然没有一次竞赛类的经历&#xff0c;也没有拿得出手的项目&#xff0c;我陷入了深深的焦虑。 听说蓝桥杯的…

[架构之路-183]-《软考-系统分析师》-13-系统设计 - 高内聚低耦合详解、图解以及技术手段

目录 第1章 什么是高内聚低耦合 1.1 概念 1.2 目的 1.3 什么时候需要进行高内聚低耦合 1.4 什么系统需要关注高内聚、低耦合 第2章 分类 2.1 内聚的分类 2.2 耦合的分类 第3章 增加高内聚降低耦合度的方法 3.1 增加高内聚 3.2 降低耦合度 第1章 什么是高内聚低耦…

超详细的R语言svykm函数绘制复杂抽样设计数据cox回归生存曲线(Kaplan-Meier)

我们在既往的文章《R语言绘制复杂抽样设计数据cox回归生存曲线(Kaplan-Meier)》中介绍了怎么使用jskm包的svykm函数绘制复杂抽样设计数据cox回归生存曲线(Kaplan-Meier)&#xff0c;但是有粉丝觉得讲得不够详细&#xff0c;希望讲得详细一点&#xff0c;今天我们继续来介绍一下…

排序算法 — 归并排序

文章目录 归并排序介绍从下往上的归并排序从上往下的归并排序 归并排序实现从上往下的归并排序从下往上的归并排序 归并排序的时间复杂度和稳定性归并排序时间复杂度归并排序稳定性 代码实现核心&总结 每日一道算法&#xff0c;提高脑力。第五天(时隔7天&#xff0c;终于回…

Mybatis 框架 ( 一 ) 基本步骤

1.概念 1.1.什么是Mybatis框架 &#xff08;1&#xff09;Mybatis是一个半ORM&#xff08;Object Relation Mapping 对象关系映射&#xff09;框架&#xff0c;它内部封装了JDBC&#xff0c;开发时只需要关注SQL语句本身&#xff0c;不需要花费精力去处理加载驱动、创建连接、…

【工具使用】- git实现gitee托管代码以及检出代码

1. 下载Git工具 git下载地址1&#xff1a;https://git-scm.com/download/win git下载2&#xff1a;https://mirrors.tuna.tsinghua.edu.cn/github-release/git-for-windows/git/Git%20for%20Windows%202.40.1/ 下载完成后安装 安装直接执行exe可执行程序&#xff0c;下一步…

Packet Tracer - 配置 RIPv2

Packet Tracer - 配置 RIPv2 目标 第 1 部分&#xff1a;配置 RIPv2 第 2 部分&#xff1a;验证配置 拓扑图 背景信息 尽管在现代网络中极少使用 RIP&#xff0c;但是作为了解基本网络路由的基础则十分有用。 在本活动中&#xff0c;您将使用适当的网络语句和被动接口配置…

【Java笔试强训 24】

&#x1f389;&#x1f389;&#x1f389;点进来你就是我的人了博主主页&#xff1a;&#x1f648;&#x1f648;&#x1f648;戳一戳,欢迎大佬指点! 欢迎志同道合的朋友一起加油喔&#x1f93a;&#x1f93a;&#x1f93a; 目录 一、选择题 二、编程题 &#x1f525;年终奖 …

VC++ | MFC应用程序设计:框架搭建

VC | MFC应用程序设计&#xff1a;框架搭建 时间&#xff1a;2023-05-01 文章目录 VC | MFC应用程序设计&#xff1a;框架搭建1.启动程序2.新建项目2-1.新建项目2-2.应用程序类型2-3.文档模板属性2-4.用户界面功能2-5.高级功能选项2-6.生成的类2-7.解决方案资源管理器 3.工程文…

springboot websocket通信

目录 一、websocket是什么 二、实现websocket 2.1参考学习b站资料&#xff08;一定要看&#xff0c;前后端详细&#xff09; 2.2学习配套代码 一、websocket是什么 WebSocket_ohana&#xff01;的博客-CSDN博客 二、实现websocket 2.1参考学习b站资料&#xff08;一定要看…