PHP手动解析JWT令牌教程
时间:2026-05-08 20:27:01 461浏览 收藏
PHP中手动解析JWT令牌绝非简单调用JWT::decode()即可完成——该方法仅负责解码与基础解析,完全不校验签名有效性、算法合规性及时间字段(如exp、nbf),若忽略显式传入密钥$key、严格限定$allowed_algs数组并手动验证过期与生效时间,将导致严重安全漏洞:攻击者可轻松伪造管理员权限token、绕过签名检查、滥用已过期凭证;尤其在v6.4+版本中,$allowed_algs已成为不可省略的强制参数,且RS256等非对称算法还需正确加载PEM公钥资源,任何一步疏漏都可能让系统“裸奔”于未授权访问风险之中。

PHP中用 firebase/php-jwt 手动验证JWT签名是否有效
直接说结论:不能只调用 JWT::decode() 就算完事,它默认不校验签名,也不检查过期时间,必须显式传入 $key 和 $allowed_algs,否则等于裸奔。
常见错误现象是:伪造的 token(比如把 payload 改成 {"admin":true} 后重签一个无效签名)也能 decode 成功,只是返回数组而已——因为没校验签名,decode() 本质只是 base64url 解码 + JSON 解析。
- 必须传入真实密钥(
$key),且类型匹配:HS256 用字符串,RS256 用 PEM 公钥资源或字符串 - 必须指定
$allowed_algs数组,比如['HS256'],否则会跳过算法校验(CVE-2015-2951 类风险) - 过期(
exp)、生效时间(nbf)、签发者(iss)等都得自己检查,decode()不自动做
JWT::decode() 报错 DomainException: Algorithm not allowed
这是最常卡住人的点:不是密钥错了,而是没传 $allowed_algs 参数,或者传了但内容不匹配 header 中的 alg 字段。
比如 token header 是 {"alg":"HS256"},但你写的是 ['RS256'],就会报这个错;更隐蔽的是漏传该参数——函数签名里它是第 3 个必填参数(v6+),旧版 v5 可能容忍空数组,但行为不可靠。
- v6.4+ 版本中,
JWT::decode($jwt, $key, $allowed_algs)第三个参数不可省略 - 别用
['none']或空数组绕过,那等于关掉签名校验 - 如果 token 来自不同服务混用多种算法(极不推荐),才考虑传
['HS256', 'RS256'],但务必确保密钥类型和算法严格对应
手动验证 exp、nbf 等时间字段的必要性
JWT::decode() 返回的是原始 payload 数组,它不会主动抛异常或过滤过期 token。你拿到 $decoded->exp 是个 Unix 时间戳,但 PHP 不会自动比对 time()。
典型翻车场景:用户登出后 token 没进黑名单,又没检查 exp,结果过期 token 还能访问敏感接口。
- 必须手动判断:
if (isset($decoded->exp) && $decoded->exp nbf(not before)同理,$decoded->nbf > time()就该拒绝- 注意时区:所有时间戳都是 UTC,别用
date('U')以外的方式生成或比较
用 openssl_pkey_get_public() 加载 RS256 公钥的坑
如果你用 RSA 签名(比如 Auth0、AWS Cognito),$key 参数不能直接传 PEM 字符串,得先转成资源,否则 decode() 会静默失败或报错 Signature verification failed。
错误写法:JWT::decode($token, "-----BEGIN PUBLIC KEY-----...", ['RS256']) —— 这在 v6+ 会直接抛异常。
- 正确做法:用
$pubKey = openssl_pkey_get_public($pemString),再传$pubKey给decode() - 记得检查返回值:
if (!$pubKey) { throw new Exception('Invalid public key'); } - PEM 字符串必须包含完整头尾(
-----BEGIN PUBLIC KEY-----),且换行符是 \n(Windows 下可能多 \r 导致加载失败)
decode() 不等于“验证”,它只是解码入口;真正的验证是密钥 + 算法 + 时间字段三者缺一不可的手动组合检查。今天关于《PHP手动解析JWT令牌教程》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
396 收藏
-
318 收藏
-
358 收藏
-
278 收藏
-
126 收藏
-
470 收藏
-
461 收藏
-
426 收藏
-
472 收藏
-
132 收藏
-
152 收藏
-
183 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习