朝花夕拾-后端-golang&mangodb-登录登出token失效

284 阅读2分钟

在线体验

shouye.png

设计

登录逻辑

登录

业务请求token拦截逻辑

拦截

登出逻辑

登出

技术预演

数据库

  • redis比mangodb更适合在这样的高频读写场景
  • 网站初期访问量是个位数级别的
  • mangodb的TTL索引功能可以实现文档数据指定失效时间

golang

  • mongo-driver是一个较高使用的包
  • gin框架也是在golang的web框架中比较知名
  • gin的中间件方法在当前项目机制下面能够很好的实现token拦截以及登录控制

jwt

  • 使用jwt的token方案也是比较成熟的一种机制

具体实现

数据库

  • 创建失效索引
db.token.createIndex( { "expiredate": 1 }, { expireAfterSeconds: 0 } )

gin

  • 中间件的使用
func InitRouter() *gin.Engine {
    r := gin.New()
    r.Use(auth.Auth())
}
  • token失效中间
func Auth() gin.HandlerFunc {
	return func(c *gin.Context) {
		Token := c.GetHeader("Token")
		fmt.Println("Token", Token)
		if Token != "" {
			appG := app.Gin{C: c}
			filter := bson.D{{"token", Token}}
			mg := database.NewMgo("token")
			var tokenInfo InterfaceEntity.TokenInfo
			err := database.FindOne(mg, filter).Decode(&tokenInfo)
			if err == mongo.ErrNoDocuments {
				appG.Response(http.StatusUnauthorized, e.INVALID_AUTH, nil)
				c.Abort()
			} else {
				c.Next()
			}
		} else {
			c.Next()
		}
	}
}

jwt

  • 具体实现
/*
 * @Description:
 * @version: 1.0.0
 * @Author: 吴文周
 * @Date: 2021-08-28 14:47:00
 * @LastEditors: 吴文周
 * @LastEditTime: 2021-10-07 17:40:57
 */
package jwt

import (
	"fmt"
	"time"

	"github.com/dgrijalva/jwt-go"
	"github.com/fodelf/cssbattle/models/InterfaceEntity"
)

//自定义Claims
type CustomClaims struct {
	UserName string
	UserId   string
	UserIcon string
	jwt.StandardClaims
}

const (
	SECRETKEY = "woshidashuaige" //私钥
)

// 生成token
func GenerateToken(user InterfaceEntity.UserInfo) (string, error) {
	maxAge := 60 * 60 * 24 * 7
	fmt.Println("userid", user.Id)
	customClaims := &CustomClaims{
		UserName: user.UserName, //用户姓名
		UserIcon: user.UserIcon,
		UserId:   user.Id,
		StandardClaims: jwt.StandardClaims{
			ExpiresAt: time.Now().Add(time.Duration(maxAge) * time.Second).Unix(), // 过期时间,必须设置
			Issuer:    user.UserName,                                              // 非必须,也可以填充用户名,
		},
	}
	//采用HMAC SHA256加密算法
	token := jwt.NewWithClaims(jwt.SigningMethodHS256, customClaims)
	tokenString, err := token.SignedString([]byte(SECRETKEY))
	if err != nil {
		return "", err
	} else {
		return tokenString, err
	}
}

//解析token
func ParseToken(tokenString string) (*CustomClaims, error) {
	token, err := jwt.ParseWithClaims(tokenString, &CustomClaims{}, func(token *jwt.Token) (interface{}, error) {
		if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok {
			return nil, fmt.Errorf("Unexpected signing method: %v", token.Header["alg"])
		}
		return []byte(SECRETKEY), nil
	})
	if claims, ok := token.Claims.(*CustomClaims); ok && token.Valid {
		return claims, nil
	} else {
		return nil, err
	}
}

总结

  • 完备的技术选型和方案是开发的基础,在开发之前逻辑设计和技术预演尤为重要。
  • 就地取材的方式其实并不可取,在长期来看还是应该关注项目周期整个过程,技术方案的远期规划和近期选择的决断要清晰。