当前位置: 首页 > news >正文

【Go】十四、图形验证码、短信验证码、注册接口与redis的简单使用

图形验证码

如何嵌入图形验证码工作:

这里选择使用captcha 开源库进行验证码设计:

选用下面的地址进行验证码开发工作

https://zh.mojotv.cn/go/refactor-base64-captcha

基础功能构建

在 api 目录下创建 captcha.go 用来编写验证码操作

package apiimport ("github.com/gin-gonic/gin""github.com/mojocn/base64Captcha""go.uber.org/zap""net/http"
)// 使用内存方式存储验证码信息
var store = base64Captcha.DefaultMemStorefunc GetCaptcha(ctx *gin.Context) {// 配置验证码参数,数字、高:80、宽:240,长度:5、倾斜程度:0.7、背景圆圈数量:80driver := base64Captcha.NewDriverDigit(80, 240, 5, 0.7, 80)// 结合参数与生成方式,创建验证码生成器cp := base64Captcha.NewCaptcha(driver, store)id, b64s, _, err := cp.Generate()if err != nil {zap.S().Errorf("生成验证码错误:%s", err.Error())ctx.JSON(http.StatusInternalServerError, gin.H{"msg": "验证码生成错误",})}// 若没有发生错误:将图片id 和 图片的 base64 编码作为参数传递ctx.JSON(http.StatusOK, gin.H{"captchaId": id,"picPath": b64s,})
}

功能嵌入路由

将验证码放入路由:

在 router 目录下创建文件:base.go:

package apiimport ("github.com/gin-gonic/gin""github.com/mojocn/base64Captcha""go.uber.org/zap""net/http"
)// 使用内存方式存储验证码信息
var store = base64Captcha.DefaultMemStorefunc GetCaptcha(ctx *gin.Context) {// 配置验证码参数,数字、高:240、宽:80,长度:5、倾斜程度:0.7、背景圆圈数量:80driver := base64Captcha.NewDriverDigit(80, 240, 5, 0.7, 80)// 结合参数与生成方式,创建验证码生成器cp := base64Captcha.NewCaptcha(driver, store)id, b64s, _, err := cp.Generate()if err != nil {zap.S().Errorf("生成验证码错误:%s", err.Error())ctx.JSON(http.StatusInternalServerError, gin.H{"msg": "验证码生成错误",})}// 若没有发生错误:将图片id 和 图片的 base64 编码作为参数传递ctx.JSON(http.StatusOK, gin.H{"captchaId": id,"picPath":   b64s,})
}

将base 的router 配置到初始化的 Initialize 目录中的 router 初始化中:

func Routers() *gin.Engine {Router := gin.Default()// 配置跨域的拦截器Router.Use(middlewares.Cors())ApiGroup := Router.Group("/u/v1")router2.InitUserRouter(ApiGroup)router2.InitBaseRouter(ApiGroup)return Router
}

我们可以在前端直接做一个展示:

<img src="" alt="">

将验证码验证逻辑添加到登录逻辑中:

修改 form/user.go 添加验证码逻辑

package forms// 这里要注意 binding 内部的参数不可以加空格
// 请求中需要有 手机号、密码、验证码信息type PassWordLoginForm struct {Mobile     string `form:"mobile" json:"mobile" binding:"required,mobile"`PassWord   string `form:"password" json:"password" binding:"required,min=3,max=10"`Captcha    string `form:"captcha" json:"captcha" binding:"required,min=5,max=5"`Captcha_id string `form:"captcha_id" json:"captcha_id" binding:"required"`
}

再对 api/user.go 中的登录方法进行改造:

	// 绑定请求参数passwordLoginForm := forms.PassWordLoginForm{}if err := c.ShouldBind(&passwordLoginForm); err != nil {HandleValidatorError(c, err)}// 下面是新增部分,再请求参数绑定后 进行验证码验证逻辑// 直接使用 store 进行验证码验证,由于store 是在同一个 package 下的,所以该变量也可以直接使用// 若验证不正确则报错if !store.Verify(passwordLoginForm.CaptchaId, passwordLoginForm.Captcha, true) {c.JSON(http.StatusBadRequest, gin.H{"captcha": "验证码错误",})return}

之后POST访问登录:

BODY:
{"mobile": "13001350015","password": "admin123","captchaId": "y2uaAcBujzdUXz0DgX6b","captcha": "56570"
}

短信验证码

阿里云短信接口

稍后在 api 目录下进行开发,构建 api/sms.go 用来处理短信验证码的核心逻辑:

package apiimport ("fmt""github.com/aliyun/alibaba-cloud-sdk-go/sdk/requests""github.com/aliyun/alibaba-cloud-sdk-go/services/dysmsapi""github.com/gin-gonic/gin"
)func SendSms(ctx *gin.Context) {client, err := dysmsapi.NewClientWithAccessKey("cn-beijing", 你的acessKey, 你的acessSecret)if err != nil {panic(err)}request := requests.NewCommonRequest()request.Method = "POST"request.Scheme = "https"request.Domain = "dysmsapi.aliyuncs.com"request.Version = "2017-05-25"	// 必须写这个日期request.ApiName = "SendSms"// 直接写,无需在阿里云上配置request.QueryParams["RegionId"] = "cn-beijing"request.QueryParams["PhoneNumbers"] = 发送的手机号// 这里要写 签名的名字,注意是名字request.QueryParams["SignName"] = 你的签名名称// 这里要写模版的 id,注意是 idrequest.QueryParams["TemplateCode"] = 你的模版id// 参照你的模版发送请求,注意中间的验证码是可以自定义生成的request.QueryParams["TemplateParam"] = "{\"code\":" + "77777" + "}"response, err := client.ProcessCommonRequest(request)fmt.Println(client.DoAction(request, response))if err != nil {fmt.Println(err.Error())}fmt.Printf("response is %#v\n", response)
}

添加随机生成验证码的逻辑:
api/sms.go:

package apiimport ("fmt""github.com/aliyun/alibaba-cloud-sdk-go/sdk/requests""github.com/aliyun/alibaba-cloud-sdk-go/services/dysmsapi""github.com/gin-gonic/gin""math/rand""strings""time"
)// 随机验证码的生成
func GenerateSmsCode(width int) string {numeric := [10]byte{0,1,2,3,4,5,6,7,8,9}r := len(numeric)// 随机数种子rand.Seed(time.Now().Unix())var sb strings.Builderfor i := 0; i < width; i++ {fmt.Fprintf(&sb, "%d", numeric[rand.Intn(r)])}return sb.String()
}func SendSms(ctx *gin.Context) {client, err := dysmsapi.NewClientWithAccessKey("cn-beijing", "LTAI5tHRZ7GqMXx271nxuSgd", "tJErGmml9ArkhJmMoo0QK1BaJnlCWM")if err != nil {panic(err)}request := requests.NewCommonRequest()request.Method = "POST"request.Scheme = "https"request.Domain = "dysmsapi.aliyuncs.com"request.Version = "2017-05-25"request.ApiName = "SendSms"// 直接写,无需在阿里云上配置request.QueryParams["RegionId"] = "cn-beijing"request.QueryParams["PhoneNumbers"] = "13001350015"// 这里要写 签名的名字,注意是名字request.QueryParams["SignName"] = "清河个人博客网站"// 这里要写模版的 id,注意是 idrequest.QueryParams["TemplateCode"] = "SMS_461975751"// 参照你的模版发送请求,注意中间的验证码是可以自定义生成的request.QueryParams["TemplateParam"] = "{\"code\":" + GenerateSmsCode(6) + "}"response, err := client.ProcessCommonRequest(request)fmt.Println(client.DoAction(request, response))if err != nil {fmt.Println(err.Error())}fmt.Printf("response is %#v\n", response)// TODO 将验证码保存到 redis 以手机号为 key, 以验证码为 value}

验证码redis存储

虚拟机拉取 redis:

docker run -d --name redis -p 6379:6379 \
-v /mydata/redis:/usr/local/etc/redis \
redis

redis 默认数据库有 16 个 (0 - 15)

默认连接会连接到第一个 0 的 redis 数据库

将短信中的信息修改为从配置文件中读取:

在 config/config,go 中进行添加需要的配置信息:

package configtype UserSrvConfig struct {Host string `mapstructure:"host"`Port int32  `mapstructure:"port"`
}type ServerConfig struct {Name        string        `mapstructure:"name"`Port        int32         `mapstructure:"port"`UserSrvInfo UserSrvConfig `mapstructure:"user_srv"`JWTInfo     JWTConfig     `mapstructure:"jwt"`AliSmsInfo  AliSmsConfig  `mapstructure:"sms"`RedisInfo   RedisConfig   `mapstructure:"redis"`
}type JWTConfig struct {SigningKey string `mapstructure:"key"`
}// redis 配置
type RedisConfig struct {Host   string `mapstructure:"host"`Port   uint   `mapstructure:"port"`Expire int    `mapstructure:"expire"`
}// 阿里云信息配置
type AliSmsConfig struct {ApiKey    string `mapstructure:"key"`ApiSecret string `mapstructure:"secret"`
}

配置文件的修改不再显示,这里显示修改之后的核心代码:

package apiimport ("context""fmt""math/rand""mxshop-api/user-web/forms""net/http""strings""time""github.com/aliyun/alibaba-cloud-sdk-go/sdk/requests""github.com/aliyun/alibaba-cloud-sdk-go/services/dysmsapi""github.com/gin-gonic/gin""github.com/go-redis/redis/v8""mxshop-api/user-web/global"
)// 随机验证码的生成
func GenerateSmsCode(width int) string {numeric := [10]byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}r := len(numeric)// 随机数种子rand.Seed(time.Now().Unix())var sb strings.Builderfor i := 0; i < width; i++ {fmt.Fprintf(&sb, "%d", numeric[rand.Intn(r)])}return sb.String()
}func SendSms(ctx *gin.Context) {// 表单验证 要求必须传入手机号和 验证码类型sendSmsForm := forms.SendSmsForm{}if err := ctx.ShouldBind(&sendSmsForm); err != nil {HandleValidatorError(ctx, err)return}client, err := dysmsapi.NewClientWithAccessKey("cn-beijing", global.ServerConfig.AliSmsInfo.ApiKey, global.ServerConfig.AliSmsInfo.ApiSecret)smsCode := GenerateSmsCode(6)if err != nil {panic(err)}request := requests.NewCommonRequest()request.Method = "POST"request.Scheme = "https"request.Domain = "dysmsapi.aliyuncs.com"request.Version = "2017-05-25"request.ApiName = "SendSms"// 直接写,无需在阿里云上配置request.QueryParams["RegionId"] = "cn-beijing"request.QueryParams["PhoneNumbers"] = sendSmsForm.Mobile// 这里要写 签名的名字,注意是名字request.QueryParams["SignName"] = "清河个人博客网站"// 这里要写模版的 id,注意是 idrequest.QueryParams["TemplateCode"] = "SMS_461975751"// 参照你的模版发送请求,注意中间的验证码是可以自定义生成的request.QueryParams["TemplateParam"] = "{\"code\":" + smsCode + "}"response, err := client.ProcessCommonRequest(request)fmt.Println(client.DoAction(request, response))if err != nil {fmt.Println(err.Error())}fmt.Printf("response is %#v\n", response)// 将验证码保存到 redis 以手机号为 key, 以验证码为 valuerdb := redis.NewClient(&redis.Options{Addr: fmt.Sprintf("%s:%d", global.ServerConfig.RedisInfo.Host, global.ServerConfig.RedisInfo.Port),})rdb.Set(context.Background(), sendSmsForm.Mobile, smsCode, time.Duration(global.ServerConfig.RedisInfo.Expire)*time.Second)ctx.JSON(http.StatusOK, gin.H{"msg": "验证码发送成功",})
}

将验证码存储进 redis

拉取 redis 的 go 语言库:go-redis:

package apiimport ("context""fmt""math/rand""strings""time""github.com/aliyun/alibaba-cloud-sdk-go/sdk/requests""github.com/aliyun/alibaba-cloud-sdk-go/services/dysmsapi""github.com/gin-gonic/gin""github.com/go-redis/redis/v8""mxshop-api/user-web/global"
)// 随机验证码的生成
func GenerateSmsCode(width int) string {numeric := [10]byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}r := len(numeric)// 随机数种子rand.Seed(time.Now().Unix())var sb strings.Builderfor i := 0; i < width; i++ {fmt.Fprintf(&sb, "%d", numeric[rand.Intn(r)])}return sb.String()
}func SendSms(ctx *gin.Context) {client, err := dysmsapi.NewClientWithAccessKey("cn-beijing", global.ServerConfig.AliSmsInfo.ApiKey, global.ServerConfig.AliSmsInfo.ApiSecret)mobile := "13001350015"smsCode := GenerateSmsCode(6)if err != nil {panic(err)}request := requests.NewCommonRequest()request.Method = "POST"request.Scheme = "https"request.Domain = "dysmsapi.aliyuncs.com"request.Version = "2017-05-25"request.ApiName = "SendSms"// 直接写,无需在阿里云上配置request.QueryParams["RegionId"] = "cn-beijing"request.QueryParams["PhoneNumbers"] = mobile// 这里要写 签名的名字,注意是名字request.QueryParams["SignName"] = "清河个人博客网站"// 这里要写模版的 id,注意是 idrequest.QueryParams["TemplateCode"] = "SMS_461975751"// 参照你的模版发送请求,注意中间的验证码是可以自定义生成的request.QueryParams["TemplateParam"] = "{\"code\":" + smsCode + "}"response, err := client.ProcessCommonRequest(request)fmt.Println(client.DoAction(request, response))if err != nil {fmt.Println(err.Error())}fmt.Printf("response is %#v\n", response)// 将验证码保存到 redis 以手机号为 key, 以验证码为 valuerdb := redis.NewClient(&redis.Options{Addr: fmt.Sprintf("%s:%d", global.ServerConfig.RedisInfo.Host, global.ServerConfig.RedisInfo.Port),})rdb.Set(context.Background(), mobile, smsCode, 30*time.Second)ctx.JSON(http.StatusOK, gin.H{"msg": "验证码发送成功",})
}

另外,该接口还涉及到 form 表单问题,要求手机号必须填写:

在 form 目录下新建一个 go 文件:form/sms.go

package forms// 这里要注意 binding 内部的参数不可以加空格
// 请求中需要有 手机号、密码、验证码输入信息、验证码图片信息
type SendSmsForm struct {Mobile string `form:"mobile" json:"mobile" binding:"required,mobile"`// 这里为了区分开登录和注册的逻辑,包括具体场景下会出现的 找回密码等逻辑,需要添加一个 Type 表单Type uint `form:"type" json:"type" binding:"required,oneof=1 2"`
}

之后在 核心逻辑中添加表单验证的代码:

func SendSms(ctx *gin.Context) {// 表单验证 要求必须传入手机号和 验证码类型sendSmsForm := forms.SendSmsForm{}if err := ctx.ShouldBind(&sendSmsForm); err != nil {HandleValidatorError(ctx, err)return}client, err := dysmsapi.NewClientWithAccessKey("cn-beijing", global.ServerConfig.AliSmsInfo.ApiKey, global.ServerConfig.AliSmsInfo.ApiSecret)smsCode := GenerateSmsCode(6)......

在router/base.go 中添加发送短信验证码的路由:

package routerimport ("github.com/gin-gonic/gin""mxshop-api/user-web/api"
)func InitBaseRouter(Router *gin.RouterGroup) {BaseRouter := Router.Group("base"){BaseRouter.GET("captcha", api.GetCaptcha)// TODOBaseRouter.POST("send_sms", api.SendSms)}
}

之后,手机会收到验证码,验证码也会被存储在 redis 中

注册接口

创建核心逻辑的方法:

api/user.go

func Register(ctx *gin.Context) {// 配置传入的表单
}

在 forms / user.go 中添加表单验证信息:

type RegisterForm struct {Mobile   string `form:"mobile" json:"mobile" binding:"required,mobile"`Password string `form:"password" json:"password" binding:"required,min=3,max=10"`Code     string `form:"code" json:"code" binding:"required,min=6,max=6"`
}

之后继续编写注册的核心逻辑

api/user.go

func Register(ctx *gin.Context) {registerForm := forms.RegisterForm{}// 配置传入的表单,进行表单验证if err := ctx.ShouldBind(&registerForm); err != nil {HandleValidatorError(ctx, err)return}// 进行短信验证码校验rdb := redis.NewClient(&redis.Options{Addr: fmt.Sprintf("%s:%d", global.ServerConfig.RedisInfo.Host, global.ServerConfig.RedisInfo.Port),})if value, err := rdb.Get(context.Background(), registerForm.Mobile).Result(); err == redis.Nil {zap.S().Debug("key 不存在")ctx.JSON(http.StatusBadRequest, gin.H{"code": "验证码过期或未发送验证码",})return} else if value != registerForm.Code {ctx.JSON(http.StatusBadRequest, gin.H{"code": "验证码错误",})return}// 用户拨号连接 grpc 服务userConn, err := grpc.Dial(fmt.Sprintf("%s:%d", global.ServerConfig.UserSrvInfo.Host, global.ServerConfig.UserSrvInfo.Port), grpc.WithInsecure())if err != nil {zap.S().Errorw("[Register] 连接用户服务器失败", "msg", err.Error())}// 生成用户模块的 grpc 接口并调用userSrvClient := proto.NewUserClient(userConn)user, err := userSrvClient.CreateUser(context.Background(), &proto.CreateUserInfo{Nickname: registerForm.Mobile,Password: registerForm.Password,Mobile:   registerForm.Mobile,})if err != nil {zap.S().Errorf("[Register] 创建用户 失败")HandleGrpcErrorToHttp(err, ctx)return}// 注册成功,自动登录,生成登录的ToKEN// 登录成功,生成TOKENj := middlewares.NewJWT() // 获得签名// 构建传递的信息以及签名信息claims := models.CustomClaims{ID:          uint(user.Id),NickName:    user.NickName,AuthorityID: uint(user.Role),StandardClaims: jwt.StandardClaims{ // 签名相关信息NotBefore: time.Now().Unix(),               // 签名生效时间:现在ExpiresAt: time.Now().Unix() + 60*60*24*30, // 签名过期时间,从现在开始一个月Issuer:    "BaiLu",                         // 签名 对象 (公司)},}// 真正创建 TOKEN:token, err := j.CreateToken(claims)if err != nil {ctx.JSON(http.StatusInternalServerError, gin.H{"msg": "TOKEN生成失败",})return}ctx.JSON(http.StatusOK, gin.H{"id":         user.Id,"nick_name":  user.NickName,"token":      token,"expired_at": (time.Now().Unix() + 60*60*24*30) * 1000,})}

将 Register 添加到 Router 中

router/user.go

func InitUserRouter(Router *gin.RouterGroup) {// 这样就需要 /user/list 才可以进行访问了UserRouter := Router.Group("user"){// 在这里添加拦截器的作用响应位置UserRouter.GET("list", middlewares.JWTAuth(), middlewares.IsAdminAuth(), api.GetUserList)//UserRouter.GET("list", api.GetUserList)UserRouter.POST("pwd_login", api.PassWordLogin)UserRouter.POST("register", api.Register)}
}

相关文章:

  • 单片机练习题3
  • 每日优秀影视分享❗❗
  • WPF文本绑定显示格式StringFormat设置-特殊格式时间日期和多数据绑定
  • 原生dom操作快速写入html渲染(insertAdjacentHTML)
  • Cadence:Conformal系列形式验证工具
  • 深入解析Netty的Reactor模型及其实现:详解与代码示例
  • Pikachu靶场--XSS
  • excel数据透视
  • Ubuntu常见命令解释
  • 修改主频睡眠模式停止模式待机模式
  • 第五章重采样方法
  • 牛顿迭代法(求解整数的近似平方根)
  • 网络爬虫中selenium和requests这两个工具有什么区别呢?
  • 力扣爆刷第153天之TOP100五连刷(接雨水、环形链表、最长上升子序列)
  • Golang笔记:使用serial包进行串口通讯
  • [分享]iOS开发 - 实现UITableView Plain SectionView和table不停留一起滑动
  • [译]如何构建服务器端web组件,为何要构建?
  • Docker 笔记(1):介绍、镜像、容器及其基本操作
  • js 实现textarea输入字数提示
  • MQ框架的比较
  • React Transition Group -- Transition 组件
  • spark本地环境的搭建到运行第一个spark程序
  • windows下mongoDB的环境配置
  • 包装类对象
  • 聊聊springcloud的EurekaClientAutoConfiguration
  • 名企6年Java程序员的工作总结,写给在迷茫中的你!
  • 前端技术周刊 2019-02-11 Serverless
  • 实现简单的正则表达式引擎
  • 使用 Node.js 的 nodemailer 模块发送邮件(支持 QQ、163 等、支持附件)
  • 微服务核心架构梳理
  • 我的面试准备过程--容器(更新中)
  • 小李飞刀:SQL题目刷起来!
  • 一道面试题引发的“血案”
  • 正则学习笔记
  • ​ubuntu下安装kvm虚拟机
  • ‌内网穿透技术‌总结
  • #07【面试问题整理】嵌入式软件工程师
  • (04)Hive的相关概念——order by 、sort by、distribute by 、cluster by
  • (2024,LoRA,全量微调,低秩,强正则化,缓解遗忘,多样性)LoRA 学习更少,遗忘更少
  • (C++)栈的链式存储结构(出栈、入栈、判空、遍历、销毁)(数据结构与算法)
  • (Charles)如何抓取手机http的报文
  • (k8s)kubernetes 部署Promehteus学习之路
  • (MTK)java文件添加简单接口并配置相应的SELinux avc 权限笔记2
  • (回溯) LeetCode 40. 组合总和II
  • (七)glDrawArry绘制
  • (七)MySQL是如何将LRU链表的使用性能优化到极致的?
  • (三维重建学习)已有位姿放入colmap和3D Gaussian Splatting训练
  • (学习总结)STM32CubeMX HAL库 学习笔记撰写心得
  • (转)LINQ之路
  • .gitignore文件—git忽略文件
  • .NET 8.0 中有哪些新的变化?
  • .net CHARTING图表控件下载地址
  • .NET CORE 2.0发布后没有 VIEWS视图页面文件
  • .NET Core SkiaSharp 替代 System.Drawing.Common 的一些用法
  • .NET 程序如何获取图片的宽高(框架自带多种方法的不同性能)