Gin 应用多实例部署session问题、session参数与刷新

Gin 应用多实例部署session问题、session参数与刷新

一、Gin Session 存储的实现方案

  • cookie:基于cookie的实现,不安全,一般不会使用。
  • gorm:基于 GORM 的实现
  • memcached:基于 Memcached 的实现
  • memstore:基于内存的实现,一般单实例部署用的比较多,或者本地测试。
  • mongo:基于 MongoDB 的实现
  • postgres:基于 PostgreSQL 的实现
  • redis:基于 Redis 的实现,多实例部署,应该无脑选 redis 实现。
  • tester:用于测试的实现

其实Gin 中的session 是通过github.com/gorilla/sessions实现的,只不过做了二次封装。

二、memstore:基于内存的实现

2.1 基本使用

memstoregithub.com/gin-contrib/sessions库提供的一个基于内存的Session存储后端。它将Session数据存储在应用程序的内存中,适用于小型应用或用于开发和测试目的。以下是一个使用memstore的简单示例:

  1. 安装gin-contrib/sessions库:

    go get -u github.com/gin-contrib/sessions
    
  2. 使用memstore存储Session:

    package main
    
    import (
     "github.com/gin-contrib/sessions"
     "github.com/gin-contrib/sessions/memstore"
     "github.com/gin-gonic/gin"
    )
    
    func main() {
     // 初始化Gin引擎
     router := gin.Default()
        // 用内存存储 session
     // 这是基于内存的实现,第一个参数是 authentication key,最好是 32 或者64 位
     //第二个参数是 encryption key
     store := memstore.NewStore([]byte("moyn8y9abnd7q4zkq2m73yw8tu9j5ixm"),[]byte("o6idlo2cb9f9pb6h46fimllw481ldebi"))
     router.Use(sessions.Sessions("mysession", store))
    
     // 路由示例
     router.GET("/set", func(c *gin.Context) {
         // 设置Session
         session := sessions.Default(c)
         session.Set("key", "value")
         session.Save()
    
         c.JSON(200, gin.H{"message": "Session set"})
     })
    
     router.GET("/get", func(c *gin.Context) {
         // 获取Session
         session := sessions.Default(c)
         value := session.Get("key")
    
         c.JSON(200, gin.H{"key": value})
     })
    
     // 启动服务
     router.Run(":8080")
    }
    
    

    在上述示例中,我们使用了memstore.NewStore创建一个基于内存的Session存储后端。记得在实际应用中,根据实际需求选择适当的Session存储后端,例如,在生产环境中可能更常见的是使用像Redis这样的持久化存储。

    请注意,memstore是基于内存的,如果应用程序重新启动,所有存储在内存中的Session数据将被清除。因此,它最适用于短期的Session需求,而不适用于长期的数据存储。在实际生产环境中,需要根据应用程序的需求选择合适的Session存储后端。

2.2 关键参数

我们通过NewStore入口进入,可以看到,官方要求传入鉴权和加密的Key对于Key的长度越长越复杂越安全。

在正常情况下,要传入两个关键参数。第一个参数是 认证密钥(authentication key),最好是 32 或者 64 位。第二个参数是 加密密钥(encryption key)。

// 用内存存储 session
// 这是基于内存的实现,第一个参数是 authentication key,最好是 32 或者64 位
//第二个参数是 encryption key
store := memstore.NewStore([]byte("moyn8y9abnd7q4zkq2m73yw8tu9j5ixm"),[]byte("o6idlo2cb9f9pb6h46fimllw481ldebi"))
// cookie 的名字叫做sessions
ser.Use(sessions.Sessions("sessions", store))// 登录校验
ser.Use(middleware.NewLoginMiddlewareBuilder().Build())

三、使用redis:多实例部署

3.1 使用redis优势

在分布式环境下(包括单例应用多实例部署),都需要确保 Session 在每一个实例上都可以访问到,而单节点只能访问当前环境的Session。

3.2 基本使用

在使用Redis作为Session存储时,你需要使用Gin框架的github.com/gin-contrib/sessions中间件,并选择一个支持Redis的Session存储后端,比如github.com/gin-contrib/sessions/redis

以下是一个基本的使用示例:

  1. 安装依赖:

    go get -u github.com/gin-contrib/sessions
    go get -u github.com/gin-contrib/sessions/redis
    
  1. 使用Redis存储Session:

    package main
    
    import (
     "github.com/gin-contrib/sessions"
     "github.com/gin-contrib/sessions/redis"
     "github.com/gin-gonic/gin"
    )
    
    func main() {
     // 初始化Gin引擎
     router := gin.Default()
    
     // 使用Redis存储Session
     store, err := redis.NewStore(
         16,               // 最大空闲链接数量,过大会浪费,过小将来会触发性能瓶颈
         "tcp",            // 指定与Redis服务器通信的网络类型,通常为"tcp"
         "localhost:6379", // Redis服务器的地址,格式为"host:port"
         "",               // Redis服务器的密码,如果没有密码可以为空字符串
         []byte("95osj3fUD7fo0mlYdDbncXz4VD2igvf0"), // authentication key
         []byte("0Pf2r0wZBpXVXlQNdpwCXN4ncnlnZSc3"), // encryption key
     )
    
     if err != nil {
         panic(err)
     }
    
     // 设置Session中间件
     router.Use(sessions.Sessions("mysession", store))
    
     // 路由示例
     router.GET("/set", func(c *gin.Context) {
         // 设置Session
         session := sessions.Default(c)
         session.Set("key", "value")
         session.Save()
    
         c.JSON(200, gin.H{"message": "Session set"})
     })
    
     router.GET("/get", func(c *gin.Context) {
         // 获取Session
         session := sessions.Default(c)
         value := session.Get("key")
    
         c.JSON(200, gin.H{"key": value})
     })
    
     // 启动服务
     router.Run(":8080")
    }
    
    

    在上述示例中,我们使用了redis.NewStore创建一个基于Redis的Session存储后端。你需要提供Redis服务器的地址、密码和密钥等信息。

四、信息安全的三个核心概念

  1. 身份认证(Authentication): 身份认证是确认用户或系统的身份是否合法的过程。在身份认证中,用户提供的身份信息(例如用户名和密码、生物特征等)被验证以确定其是否有权访问系统或资源。常见的身份认证方式包括用户名密码认证、多因素认证(例如使用手机验证码或硬件令牌)、生物特征认证等。
  2. 数据加密(Encryption): 数据加密是通过使用算法将信息转化为密文,以确保只有具备正确密钥的人或系统能够解密和访问原始信息。数据加密对于保护敏感信息、防止数据泄露和维护隐私非常重要。常见的加密算法包括对称加密(同一个密钥用于加密和解密)和非对称加密(使用一对密钥,一个用于加密,另一个用于解密)。
  3. 权限控制(Authorization): 权限控制是确保用户或系统在身份认证成功后只能访问其被授权的资源和执行其被授权的操作的过程。这包括定义用户或系统的角色、分配权限、限制访问范围等。权限控制有助于防止未经授权的访问和确保系统的安全性。

五、Gin Session 参数

5.1 参数介绍

在Gin框架中,Session的参数可以通过Options方法来传入OptionOptions方法用于配置Session的一些参数,以满足应用程序的需求。

参数详细解释:

字段 含义 示例值
Path Cookie的路径 "/"
Domain Cookie的域 "your-domain.com"
MaxAge 最大生存时间(秒) 3600
Secure 是否仅通过HTTPS传输 true
HttpOnly 是否禁止通过JavaScript访问Cookie true
SameSite SameSite属性 http.SameSiteLaxMode

举个例子:

package main

import (
    "github.com/gin-contrib/sessions"
    "github.com/gin-gonic/gin"
    "net/http"
)

func main() {
    // 初始化Gin引擎
    router := gin.Default()

    // 配置Session
    store := sessions.NewCookieStore([]byte("your-secret-key"))
    store.Options(sessions.Options{
        Path:     "/",
        Domain:   "your-domain.com",
        MaxAge:   3600, // 设置为3600秒,即1小时
        Secure:   true, // 仅通过HTTPS传输Cookie
        HttpOnly: true, // 禁止通过JavaScript访问Cookie
        SameSite: http.SameSiteLaxMode, // SameSite属性,限制在顶级导航时发送
    })
    router.Use(sessions.Sessions("mysession", store))

    // 路由示例
    router.GET("/set", func(c *gin.Context) {
        // 设置Session
        session := sessions.Default(c)
        session.Set("key", "value")
        session.Save()

        c.JSON(200, gin.H{"message": "Session set"})
    })

    router.GET("/get", func(c *gin.Context) {
        // 获取Session
        session := sessions.Default(c)
        value := session.Get("key")

        c.JSON(200, gin.H{"key": value})
    })

    // 启动服务
    router.Run(":8080")
}

六、Session 自动刷新

实现方式,在中间件中设置一个更新时间:

// LoginMiddlewareBuilder 结构体的 Build 方法,用于构建 Gin 中间件
func (l *LoginMiddlewareBuilder) Build() gin.HandlerFunc {
    // 使用 gob 注册 time.Now(),以便在 Session 中存储 time.Time 类型
    gob.Register(time.Now())

    // 返回一个 Gin 中间件函数
    return func(ctx *gin.Context) {
        // 检查当前请求路径是否为不需要登录校验的路径
        if ctx.Request.URL.Path == "/users/login" ||
            ctx.Request.URL.Path == "/users/signup" {
            // 如果是不需要登录校验的路径,直接返回,不进行后续的登录检查
            return
        }

        // 获取默认的 Session
        sess := sessions.Default(ctx)

        // 获取 Session 中存储的 userId
        id := sess.Get("userId")

        // 如果 userId 不存在,说明用户未登录,返回未授权状态码
        if id == nil {
            ctx.AbortWithStatus(http.StatusUnauthorized)
            return
        }

        // 获取 Session 中的 updateTime
        updateTime := sess.Get("update_time")

        // 设置 Session 的 userId,并配置 Session 的过期时间为 60 秒
        sess.Set("userId", id)
        sess.Options(sessions.Options{
            MaxAge: 60,
        })

        now := time.Now()

        // 如果 updateTime 为空,说明是第一次登录,设置 update_time 并保存 Session
        if updateTime == nil {
            sess.Set("update_time", now)
            if err := sess.Save(); err != nil {
                panic(err)
            }
        }

        // 如果 updateTime 不为空,说明已经登录过,检查是否超过 10 秒,超过则刷新 update_time 并保存 Session
        updateTimeVal, _ := updateTime.(time.Time)
        if now.Sub(updateTimeVal) > time.Second*10 {
            sess.Set("update_time", now)
            if err := sess.Save(); err != nil {
                panic(err)
            }
        }
    }
}

缺点:由于这种方式每次都要从Redis中读写数据,在高并发中并不适合。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容