如何嵌入图形验证码工作:
这里选择使用captcha 开源库进行验证码设计:
选用下面的地址进行验证码开发工作
https://zh.mojotv.cn/go/refactor-base64-captcha
在 api 目录下创建 captcha.go 用来编写验证码操作
package api
import (
"github.com/gin-gonic/gin"
"github.com/mojocn/base64Captcha"
"go.uber.org/zap"
"net/http"
)
// 使用内存方式存储验证码信息
var store = base64Captcha.DefaultMemStore
func GetCaptcha(ctx *gin.Context) {
// 配置验证码参数,数字、高:80、宽:240,长度:5、倾斜程度:0.7、背景圆圈数量:80
driver := 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 api
import (
"github.com/gin-gonic/gin"
"github.com/mojocn/base64Captcha"
"go.uber.org/zap"
"net/http"
)
// 使用内存方式存储验证码信息
var store = base64Captcha.DefaultMemStore
func GetCaptcha(ctx *gin.Context) {
// 配置验证码参数,数字、高:240、宽:80,长度:5、倾斜程度:0.7、背景圆圈数量:80
driver := 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 api
import (
"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,注意是 id
request.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 api
import (
"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.Builder
for 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,注意是 id
request.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:
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 config
type 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 api
import (
"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.Builder
for 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,注意是 id
request.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, 以验证码为 value
rdb := 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 api
import (
"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.Builder
for 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,注意是 id
request.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, 以验证码为 value
rdb := 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 router
import (
"github.com/gin-gonic/gin"
"mxshop-api/user-web/api"
)
func InitBaseRouter(Router *gin.RouterGroup) {
BaseRouter := Router.Group("base")
{
BaseRouter.GET("captcha", api.GetCaptcha)
// TODO
BaseRouter.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(®isterForm); 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
// 登录成功,生成TOKEN
j := 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)
}
}