golang原生实现JWT的示例代码
来源:脚本之家
2023-05-12 15:16:13
0浏览
收藏
本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《golang原生实现JWT的示例代码》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~
JWT(JSON Web Token)是一种基于JSON的安全令牌,可以用于在不同系统之间传输认证信息。在Go中实现JWT验证,可以通过标准库crypto/hmac
、crypto/sha256
和encoding/base64
来编写自己的JWT。
获取Token
我们在此封装一个JWT的struct结构体(由于除了Payload,其他很大可能不会在其他地方用到,所以不公开)
type JWT struct { header string Payload string signature string }
将base64的编码封装一下方便使用
func encodeBase64(data string) string { return base64.RawURLEncoding.EncodeToString([]byte(data)) }
我们封装一个用来生成签名的方法
func generateSignature(key []byte, data []byte) (string, error) { // 创建一个哈希对象 hash := hmac.New(sha256.New, key) // 将要签名的信息写入哈希对象中 hash.Write(data) _, err := hash.Write(data) if err != nil { return "", err } // hash.Sum()计算签名,在这里会返回签名内容 // 将签名经过base64编码生成字符串形式返回。 return encodeBase64(string(hash.Sum(nil))), nil }
我们封装一个CreateToken用于生成Token(该方法的参数key为(生成签名所使用的密钥))
func CreateToken(key []byte, payloadData any) (string, error) { // 标准头部 header := `{"alg":"HS256","typ":"JWT"}` // 将负载的数据转换为json payload, jsonErr := json.Marshal(payloadData) if jsonErr != nil { return "", fmt.Errorf("负载json解析错误") } // 将头部和负载通过base64编码,并使用.作为分隔进行连接 encodedHeader := encodeBase64(header) encodedPayload := encodeBase64(string(payload)) HeaderAndPayload := encodedHeader + "." + encodedPayload // 使用签名使用的key将传入的头部和负载连接所得的数据进行签名 signature, err := generateSignature(key, []byte(HeaderAndPayload)) if err != nil { return "", err } // 将token的三个部分使用.进行连接并返回 return HeaderAndPayload + "." + signature, nil }
解析Token
我们封装一个解析token的方法
func ParseJwt(token string, key []byte) (*JWT, error) { // 分解规定,我们使用.进行分隔,所以我们通过.进行分隔成三个字符串的数组 jwtParts := strings.Split(token, ".") // 数据数组长度不是3就说明token在格式上就不合法 if len(jwtParts) != 3 { return nil, fmt.Errorf("非法token") } // 分别拿出 encodedHeader := jwtParts[0] encodedPayload := jwtParts[1] signature := jwtParts[2] // 使用key将token中的头部和负载用.连接后进行签名 // 这个签名应该个token中第三部分的签名一致 confirmSignature, err := generateSignature(key, []byte(encodedHeader+"."+encodedPayload)) if err != nil { return nil, fmt.Errorf("生成签名错误") } // 如果不一致 if signature != confirmSignature { return nil, fmt.Errorf("token验证失败") } // 将payload解base64编码 dstPayload, _ := base64.RawURLEncoding.DecodeString(encodedPayload) // 返回我们的JWT对象以供后续使用 return &JWT{encodedHeader, string(dstPayload), signature}, nil }
实际使用
我们构造一个用户的结构体
type UserInfo struct { Name string `json:"name"` Password string `json:"password"` }
我们这次使用123456作为密钥简单的验证一下
var Key []byte = []byte("12346")
在此我们构造一个验证的中间件
func jwtConfirm(context *gin.Context) { // 登录不需要token if context.Request.RequestURI == "/login" { return } // 拿出token token := context.GetHeader("Token") // 进行解析验证 jwt, err := utils.ParseJwt(token, Key) if err != nil { context.JSON(200, gin.H{ "msg": err.Error(), }) // 有问题就流产掉(我也不知道怎么翻译好了,香蕉猫.jpg) context.Abort() } // 验证通过就将负载返回回去 context.JSON(200, gin.H{ "payload": jwt.Payload, }) }
我们在此基础上就可以使用了,这边使用gin框架简单的测试一下
func main() { // 使用默认路由 router := gin.Default() // 注册中间件 router.Use(jwtConfirm) // 简单做两个服务 router.POST("/login", func(context *gin.Context) { // 接收用户参数 var userInfo UserInfo = UserInfo{} // 使用jsonbind接收 bindErr := context.ShouldBindJSON(&userInfo) if bindErr != nil { context.JSON(200, gin.H{ "msg": bindErr.Error(), }) } // 使用密钥做出token jwt, err := utils.CreateJwt(Key, userInfo) if err != nil { fmt.Println(err) } // 我们将token直接返回用于测试 context.JSON(200, gin.H{ "token": jwt, }) }) router.GET("/doing") router.Run() }
测试结果
我们拿到了token
我们现在去试一下如果不带token的结果
我们试一下携带错误token的情况
我们最后测试一下正确的token
结语
在实际环境中会使用更复杂的情况进行使用(例如密钥会更加复杂,会在pyload中设置失效时间等等),但是生成token和解析token的操作和上述的操作差别不大
今天带大家了解了JWT的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~
版本声明
本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除

- 上一篇
- redis使用Lua脚本解决多线程下的超卖问题及原因解析

- 下一篇
- RedisDesktopManager(Redis可视化工具)安装及使用图文教程
查看更多
最新文章
-
- Golang · Go教程 | 3小时前 |
- Go监控CPU使用率技巧分享
- 132浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang端口复用与负载均衡配置
- 100浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Go语言设计模式:打造优雅代码架构
- 416浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang字符串拼接优化:strings.Builder vs +性能对比
- 198浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- GolangGOPROXY设置教程:加速依赖下载
- 214浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang自定义包与模块创建指南
- 207浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- GolangTCP服务器开发教程与通信实例
- 424浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang微服务gRPC负载均衡解析
- 249浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Go函数表与Switch性能对比解析
- 123浏览 收藏
-
- Golang · Go教程 | 4小时前 |
- Golang搭建gRPC服务:proto定义与生成详解
- 119浏览 收藏
查看更多
课程推荐
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 514次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 499次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
查看更多
AI推荐
-
- AI Mermaid流程图
- SEO AI Mermaid 流程图工具:基于 Mermaid 语法,AI 辅助,自然语言生成流程图,提升可视化创作效率,适用于开发者、产品经理、教育工作者。
- 639次使用
-
- 搜获客【笔记生成器】
- 搜获客笔记生成器,国内首个聚焦小红书医美垂类的AI文案工具。1500万爆款文案库,行业专属算法,助您高效创作合规、引流的医美笔记,提升运营效率,引爆小红书流量!
- 648次使用
-
- iTerms
- iTerms是一款专业的一站式法律AI工作台,提供AI合同审查、AI合同起草及AI法律问答服务。通过智能问答、深度思考与联网检索,助您高效检索法律法规与司法判例,告别传统模板,实现合同一键起草与在线编辑,大幅提升法律事务处理效率。
- 666次使用
-
- TokenPony
- TokenPony是讯盟科技旗下的AI大模型聚合API平台。通过统一接口接入DeepSeek、Kimi、Qwen等主流模型,支持1024K超长上下文,实现零配置、免部署、极速响应与高性价比的AI应用开发,助力专业用户轻松构建智能服务。
- 733次使用
-
- 迅捷AIPPT
- 迅捷AIPPT是一款高效AI智能PPT生成软件,一键智能生成精美演示文稿。内置海量专业模板、多样风格,支持自定义大纲,助您轻松制作高质量PPT,大幅节省时间。
- 626次使用
查看更多
相关文章
-
- Golang基于JWT与Casbin身份验证授权实例详解
- 2022-12-30 356浏览
-
- Go开发Gin项目添加jwt功能实例详解
- 2023-01-01 375浏览
-
- go语言beego框架jwt身份认证实现示例
- 2023-01-07 184浏览
-
- gogin+token(JWT)验证实现登陆验证
- 2023-01-01 296浏览
-
- golang jwt+token验证的实现
- 2023-02-21 188浏览