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

文章目录

    • 一、Gin Session 存储的实现方案
    • 二、`memstore`:基于内存的实现
      • 2.1 基本使用
      • 2.2 关键参数
    • 三、使用`redis`:多实例部署
      • 3.1 使用redis优势
      • 3.2 基本使用
    • 四、信息安全的三个核心概念
    • 五、Gin Session 参数
      • 5.1 参数介绍
    • 六、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 mainimport ("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 keystore := memstore.NewStore([]byte("moyn8y9abnd7q4zkq2m73yw8tu9j5ixm"),[]byte("o6idlo2cb9f9pb6h46fimllw481ldebi"))router.Use(sessions.Sessions("mysession", store))// 路由示例router.GET("/set", func(c *gin.Context) {// 设置Sessionsession := sessions.Default(c)session.Set("key", "value")session.Save()c.JSON(200, gin.H{"message": "Session set"})})router.GET("/get", func(c *gin.Context) {// 获取Sessionsession := 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
    
  2. 使用Redis存储Session:

    package mainimport ("github.com/gin-contrib/sessions""github.com/gin-contrib/sessions/redis""github.com/gin-gonic/gin"
    )func main() {// 初始化Gin引擎router := gin.Default()// 使用Redis存储Sessionstore, 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) {// 设置Sessionsession := sessions.Default(c)session.Set("key", "value")session.Save()c.JSON(200, gin.H{"message": "Session set"})})router.GET("/get", func(c *gin.Context) {// 获取Sessionsession := 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的一些参数,以满足应用程序的需求。

参数详细解释:

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

举个例子:

package mainimport ("github.com/gin-contrib/sessions""github.com/gin-gonic/gin""net/http"
)func main() {// 初始化Gin引擎router := gin.Default()// 配置Sessionstore := sessions.NewCookieStore([]byte("your-secret-key"))store.Options(sessions.Options{Path:     "/",Domain:   "your-domain.com",MaxAge:   3600, // 设置为3600秒,即1小时Secure:   true, // 仅通过HTTPS传输CookieHttpOnly: true, // 禁止通过JavaScript访问CookieSameSite: http.SameSiteLaxMode, // SameSite属性,限制在顶级导航时发送})router.Use(sessions.Sessions("mysession", store))// 路由示例router.GET("/set", func(c *gin.Context) {// 设置Sessionsession := sessions.Default(c)session.Set("key", "value")session.Save()c.JSON(200, gin.H{"message": "Session set"})})router.GET("/get", func(c *gin.Context) {// 获取Sessionsession := 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}// 获取默认的 Sessionsess := sessions.Default(ctx)// 获取 Session 中存储的 userIdid := sess.Get("userId")// 如果 userId 不存在,说明用户未登录,返回未授权状态码if id == nil {ctx.AbortWithStatus(http.StatusUnauthorized)return}// 获取 Session 中的 updateTimeupdateTime := sess.Get("update_time")// 设置 Session 的 userId,并配置 Session 的过期时间为 60 秒sess.Set("userId", id)sess.Options(sessions.Options{MaxAge: 60,})now := time.Now()// 如果 updateTime 为空,说明是第一次登录,设置 update_time 并保存 Sessionif updateTime == nil {sess.Set("update_time", now)if err := sess.Save(); err != nil {panic(err)}}// 如果 updateTime 不为空,说明已经登录过,检查是否超过 10 秒,超过则刷新 update_time 并保存 SessionupdateTimeVal, _ := updateTime.(time.Time)if now.Sub(updateTimeVal) > time.Second*10 {sess.Set("update_time", now)if err := sess.Save(); err != nil {panic(err)}}}
}

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/651145.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

语图奇缘:林浩然与杨凌芸的哲学漫画大冒险

语图奇缘:林浩然与杨凌芸的哲学漫画大冒险 Language Odyssey: The Philosophical Comic Adventure of Lin Haoran and Yang Lingyun 在一个充满逻辑谜题和言语陷阱的城市——逻言市,住着两位热衷于探索语言奥秘的年轻人,林浩然和杨凌芸。林浩…

Qt的绘制是如何发生的

QEvent::WindowDeactivate 当用户鼠标在应用之外时,会触发QEvent::WindowDeactivate事件, Qt5Widgetsd.dll!QApplication::setActiveWindow(QWidget * act) Line 2072 C Qt5Widgetsd.dll!QApplicationPrivate::notifyActiveWindowChange(QWindo…

一篇文章带你了解C++中隐含的this指针

文章目录 一、this指针的引出二、this指针的特性【面试题】 一、this指针的引出 我们先来定义一个日期类Date,下面这段代码执行的结果是什么呢? class Date { public:void Init(int year, int month, int day){_year year;_month month;_day day;}v…

2024新版68套Axure RP大数据可视化大屏模板及通用组件+PSD源文件

Axure RP数据可视化大屏模板及通用组件库2024新版重新制作了这套新的数据可视化大屏模板及通用组件库V2版。新版本相比于V1版内容更加丰富和全面,但依然秉承“敏捷易用”的制作理念,这套作品也同样延续着我们对细节的完美追求,整个设计制作过…

【PythonRS】Rasterio库安装+基础函数使用教程

Rasterio是一个Python库,专门用于栅格数据的读写操作。它支持多种栅格数据格式,如GeoTIFF、ENVI和HDF5,为处理和分析栅格数据提供了强大的工具。RasterIO适用于各种栅格数据应用,如卫星遥感、地图制作等。通过RasterIO&#xff0c…

Two-factor authentication (2FA) is required for your GitHub account解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

SkyWalking介绍与使用docker-compose部署服务

一、Skywalking概述 1、Skywalking介绍 Skywalking是分布式系统的应用程序性能监视工具,专为微服务,云原生架构和基于容器(Docker,K8S,Mesos)架构而设计,它是一款优秀的APM(Application Performance Management)工具,包括了分布式追踪,性能指标分析和服务依赖分析等…

gitee仓库使用中的警告

当 Git 执行 git pull 命令时,有时候会出现类似下面的警告信息: warning: ----------------- SECURITY WARNING ---------------- warning: | TLS certificate verification has been disabled! | warning: ------------------------------------------…

Python退出主程序

import sys sys.exit(1) exit函数的使用方法 Exit the interpreter by raising SystemExit(status).If the status is omitted or None, it defaults to zero (i.e., success). If the status is an integer, it will be used as the system exit status. If it is another kin…

网络安全03---Nginx 解析漏洞复现

目录 一、准备环境 二、实验开始 2.1上传压缩包并解压 2.2进入目录,开始制作镜像 2.3可能会受之前环境影响,删除即可 ​编辑 2.4制作成功结果 2.5我们的环境一个nginx一个php 2.6访问漏洞 2.7漏洞触发结果 2.8上传代码不存在漏洞 2.9补充&#…

ubuntu 编译使用 liblas 读取点云

在ubuntu上使用las读取点云 1、环境配置 1.1、安装libgeotiff 下载依赖 sudo apt-get install libtiff-dev //安装libtiff sudo apt-get install libproj-dev //安装libproj下载源码,编译 如下该是libgeotiff-1.3.0版本安装包 wget https://download.osgeo.o…

【蓝桥杯冲冲冲】旅行计划

蓝桥杯备赛 | 洛谷做题打卡day18 文章目录 蓝桥杯备赛 | 洛谷做题打卡day18旅行计划题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 提示题解代码我的一些话 旅行计划 题目描述 Kira酱要去一个国家旅游。这个国家有 N N N 个城市,编号为 1 1 1 至 N N…

计算机网络(第六版)复习提纲11

二、CSMA/CD协议 1.广播信道上的一对一通信 2.以太网的两个措施 a)采用无连接的方法,不可靠 b)发送的数据都是用曼彻斯特编码 3.CSMA/CD协议(先听后说,边听边说) a)前身是ALOHA协议,想发就发,冲突时随机等待…

SpringSecurity(16)——OAuth2客户端授权模式

工作流程 基本使用 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-security</artifactId><version>2.3.12.RELEASE</version> </dependency> <dependency><groupId>…

大数据Doris(五十九):SQL函数之字符串函数(三)

文章目录 SQL函数之字符串函数(三) 一、​​​​​​​NULL_OR_EMPTY (VARCHAR str)

力扣1312. 让字符串成为回文串的最少插入次数

动态规划 思路&#xff1a; 通过插入字符构造回文串&#xff0c;要想插入次数最少&#xff0c;可以将字符串 s 的逆序 s 进行比较找出最长公共子序列&#xff1b;可以先分析&#xff0c;字符串 s 通过插入得到回文串 ps&#xff0c;其中间的字符应该不会变化&#xff1a; 若 s…

IO复用之epoll模型

什么是epoll epoll 是 Linux 操作系统提供的一种高性能的事件通知机制&#xff0c;用于处理大量文件描述符上的事件。它是一种 I/O 事件通知机制&#xff0c;通常用于处理网络编程中的并发连接。 在传统的 I/O 模型中&#xff0c;程序通常使用 select 或 poll 函数来等待多个…

基于springboot+vue的校园资料分享平台(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战 主要内容&#xff1a;毕业设计(Javaweb项目|小程序等)、简历模板、学习资料、面试题库、技术咨询 文末联系获取 项目背景…

数据灾难恢复:应对._locked勒索病毒的有效方法

导言&#xff1a; 近年来&#xff0c;网络犯罪愈发猖獗&#xff0c;其中一种威胁备受关注——._locked勒索病毒。这种恶意软件的攻击方式主要通过加密用户的数据文件&#xff0c;随后勒索受害者以解密密钥为代价。在这篇文章中&#xff0c;我们将介绍._locked勒索病毒的特点、…

qt学习:实战 http请求获取qq的吉凶

目录 利用的api是 聚合数据 的qq号码测吉凶 编程步骤 配置ui界面 添加头文件&#xff0c;定义网络管理者和http响应槽函数 在界面的构造函数里创建管理者对象&#xff0c;关联http响应槽函数 实现按钮点击事件 实现槽函数 效果 利用的api是 聚合数据 的qq号码测吉凶 先…