- Java锁的逻辑(结合对象头和ObjectMonitor)
- 还在用饼状图?来瞧瞧这些炫酷的百分比可视化新图形(附代码实现)⛵
- 自动注册实体类到EntityFrameworkCore上下文,并适配ABP及ABPVNext
- 基于Sklearn机器学习代码实战
本文将从Web应用程序处理请求时需要用户信息,同时HTTP又是无状态协议这个矛盾点出发。从该问题出发,简单描述了解决该问题的 Token 机制,进而引出 Cookie 的实现方案.
基于此我们将详细描述 Cookie 的规范,然后详细描述具体的实现方式,进一步描述 Gin 框架对 Cookie 操作提供的 API ,最终提供了一个详细的代码实现.
我们还将详细描述 Gin 框架提供 API 的实现原理,帮助用户更好得使用这两个 API .
在 如何使用Gin搭建一个Go Web应用程序 一文中,我们已经了解了如何使用 Gin 搭建一个简单的Web应用程序。然而,在现实的Web应用程序中,大部分功能都是需要用户的身份信息才能处理。举例来说,在一个视频网站查看用户最近观看记录,如果缺少用户身份信息,此时将无法对请求进行处理.
但是HTTP协议的设计,是无状态的,也就是每次请求都是独立的。基于此,应该有一套机制,能够在用户身份认证成功后,给用户分配一个 Token ,后续用户在每次请求时,都携带上该 Token ,使得服务器能够从请求中获取用户信息,解决HTTP无状态问题。大概流程如下
上面流程中,需要服务端按照某个协议,向客户端返回 Token ;客户端通过该协议,成功解析出服务端返回的 Token ,然后在每次请求中携带该 Token 。然后服务器端再根据协议,从中解析出 Token 信息,获取请求用户信息.
当前常用的有 Cookie , Jwt , OAuth2.0 等标准,其各有优缺点。其中 Cookie 是一种存储在客户端浏览器中的数据。服务端可以通过设置HTTP响应头将 Token 存储在 Cookie 当中,并在后续请求中从 Cookie 中读取 Token 。而 JWT 则是一种基于JSON格式的安全令牌,可用于在客户端和服务端之间传递信息.
之前,我们在 一文读懂Cookie 中,已经了解 Cookie 的相关内容。基于此,我们这次使用 Cookie 来实现上述所说的流程,按照 Cookie 的规范来实现 Token 的返回和请求中 Token 的解析.
这里我们对HTTP协议中的 Cookie 规范再补充一下,这里分为两部分,第一部分是服务端如何向客户端发送 Cookie ,第二部分是客户端向服务端发送请求时如何携带 Cookie 信息.
对于服务端向客户端发送 Cookie 的手段,HTTP协议存在一个 Set-Cookie 的头部字段,服务器可以通过 Set-Cookie 头部字段将 Cookie 发送给客户端。例如下面这个例子
Set-Cookie: username=abc; expires=Wed, 09 Jun 2023 10:18:14 GMT; path=/
在这个例子中,服务器设置了一个名为 username 的 Cookie ,它的值是 abc ,过期时间是2023年6月9日,路径为 / 。浏览器在接收到该 Cookie 时,便将其保存起来.
客户端请求时携带 Cookie 的方式,则是通过HTTP协议中的 Cookie 头部字段,客户端可以通过该头部字段携带信息给服务器端,比如下面这个例子
Cookie: sessionid=1234
在这个例子中,HTTP请求中携带了一个 name 为 sessionid , value 为 1234 的 Cookie 。当服务器端接收到该 HTTP 请求后,从中解析出 Cookie 的信息,然后基于此实现后续的流程.
回看上述流程,主要分为两个大部分: 客户端和服务器端。在 客户端 部分,关键任务包括保存浏览器返回的 Cookie 信息以及在请求时携带 Cookie 信息给服务器。对于 服务器端 ,则是在通过身份校验之后,能够按照规范客户端返回 Cookie ,并在接收到请求时,能够正确解析出请求中的 Cookie 信息,识别出用户信息.
对于 客户端部分 ,在浏览器接收到HTTP响应时,如果响应体中有 Set-Cookie 头部字段,浏览器会自动保存 Cookie 信息;客户端发起请求时,需要将 Cookie 信息传递给服务器。此时浏览器会自动携带通过校验的 Cookie 。如果通过校验,此时会在HTTP请求头中携带 Cookie 信息给服务端,下面是一个大概的校验流程
在整个流程中,客户端保存 Token 信息和在请求时携带 Token 信息这两部分工作,浏览器已经帮我们实现了。剩下的工作集中在服务端的,主要涉及按照 Cookie 的规范给客户端返回用户标识,并在接收到客户端请求时从HTTP请求中读取 Cookie 以获取到用户的信息。与 Cookie 相关的详细内容可以参考文章 一文读懂Cookie .
因此下面我们需要做的两件事情,其一,服务器需要按照 Cookie 的规范往客户端发送 Cookie 的内容;其次,服务器在处理请求时,需要从HTTP请求头中读取出 Cookie 的信息,成功识别用户身份.
Gin 框架中提供了一些 API ,能够帮助我们在服务端,按照 Cookie 规范给客户端发送 Cookie 信息,同时也有 API 能够帮助我们解析 Cookie 的信息。下面我们先来了解相关的 API ,然后再基于这些 API ,搭建一个能够自动识别用户信息的 Web 应用程序.
gin.Context 对象中的 SetCookie 方法用于向客户端返回响应的同时,在 Set-Cookie 头部携带 Cookie 信息。下面是该方法的详细说明:
func (c *Context) SetCookie(name, value string, maxAge int, path, domain string, secure, httpOnly bool)
name
:cookie 的名称(必须)。 value
:cookie 的值(必须)。 maxAge
:cookie 的过期时间,以秒为单位。如果为负数,则表示会话 cookie(在浏览器关闭之后删除),如果为零,则表示立即删除 cookie(可选,默认值为-1)。 path
:cookie 的路径。如果为空字符串,则使用当前请求的 URI 路径作为默认值(可选,默认值为空字符串)。 domain
:cookie 的域名。如果为空字符串,则不设置域名(可选,默认值为空字符串)。 secure
:指定是否仅通过 HTTPS 连接发送 cookie。如果为 true,则仅通过 HTTPS 连接发送 cookie;否则,使用 HTTP 或 HTTPS 连接都可以发送 cookie(可选,默认值为 false)。 httpOnly
:指定 cookie 是否可通过 JavaScript 访问。如果为 true,则无法通过 JavaScript 访问 cookie;否则,可以通过 JavaScript 访问 cookie(可选,默认值为 true)。 在处理函数中,通过调用 SetCookie 方法,便可以向客户端发送一个HTTP cookie。这里举一个代码示例,来帮助读者更好得理解该 API ,下面举一个代码示例,如下
func main() {
router := gin.Default()
router.GET("/set-cookie", func(c *gin.Context) {
c.SetCookie("user", "john", 3600, "/", "", false, true)
c.String(http.StatusOK, "cookie set successfully")
})
router.Run(":8080")
}
在这个示例中,使用 SetCookie 方法设置一个名为 user 的 cookie。这个 cookie 的值是 john ,在 1 小时后过期。该代码还设置了路径为“/”以及HttpOnly属性为true.
下面启动该服务器,客户端向服务端发送请求,请求路径为 /set-cookie ,上面的处理函数将会被执行,然后我们来看其响应内容
# 1. 发送请求
curl -i http://localhost:8080/set-cookie
# 2. 返回响应
HTTP/1.1 200 OK
Content-Type: text/plain; charset=utf-8
Set-Cookie: user=john; Path=/; Max-Age=3600; HttpOnly
Date: Sun, 20 Aug 2023 07:39:15 GMT
Content-Length: 23
cookie set successfully
查看上面第6行,可以看到,我们通过 SetCookie 方法,成功设置了一个 Cookie ,然后以在HTTP头部的形式返回.
往客户端返回 Cookie 后,浏览器会将 Cookie 保存起来,然后在下次请求时将 Cookie 跟随请求一起发送给服务器端.
在HTTP无状态协议的情况下,我们使用 Cookie 来识别用户信息,此时服务器端需要正确解析出HTTP 头部中 Cookie 的信息, Gin 框架中的 gin.Context 提供了 Cookie 方法,方便我们获取到 Cookie 的信息。下面是该方法的定义说明
func (c *Context) Cookie(name string) (string, error)
使用 Cookie 方法可以获取指定名称的Cookie值,如果不存在指定名字的 Cookie ,此时将会返回错误。下面给一个简单示例代码的说明
func main() {
router := gin.Default()
// 定义路由
router.GET("/cookie", func(c *gin.Context) {
// 获取名为 "username" 的 cookie
cookie, err := c.Cookie("username")
if err != nil {
// 如果 cookie 不存在,则返回错误信息
c.JSON(http.StatusBadRequest, gin.H{"error": "Bad request"})
return
}
// 在响应中返回 cookie 值
c.JSON(http.StatusOK, gin.H{"username": cookie})
})
router.Run(":8080")
}
在上述示例中,我们定义了一个 /cookie 路由,使用 c.Cookie("username") 方法来获取名为 username 的 Cookie 值。如果 Cookie 不存在,则返回一个错误响应。否则,我们将在响应中返回 Cookie 的值.
下面我们通过 curl 命令来对 /cookie 请求,通过 -b 标识来携带 cookie 值
# -v, --verbose 这个参数会打开curl的详细模式,输出一些额外的信息,包括HTTP请求和响应头信息。
curl -b -v -b "username=hello cookie;" http://localhost:8080/cookie
下面我们来看具体的请求体和响应体的内容
GET /cookie HTTP/1.1
Host: localhost:8080
User-Agent: curl/7.79.1
Accept: */*
Cookie: username=hello cookie;
可以看到,我们请求体携带了 Cookie 字段, Cookie 的名称为 username ,我们前面服务器端便是尝试获取名为 username 的 Cookie,下面我们看请求的响应体,看是否成功解析了HTTP 请求 Cookie的内容
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8
Date: Sun, 20 Aug 2023 08:12:45 GMT
Content-Length: 27
{"username":"hello cookie"}
可以看到,服务端程序通过 Cookie 方法成功解析了HTTP请求头部中 Cookie 字段的值,然后将解析的结果正常返回客户端.
下面我们来搭建一个基于 Cookie 实现用户身份验证的 Web 应用程序,首先需要一个登录页面,用于验证用户身份信息,验证通过后,我们将通过 Cookie 给客户端返回一个 Token .
同时,我们还需要创建一个页面,需要验证用户身份信息,在验证过程中,我们会检查用户请求中是否携带 Cookie ,同时 Cookie 中携带的数据是否正确,基于此实现用户身份的验证。下面是一个简单代码的示例
func main() {
route := gin.Default()
route.GET("/login", func(c *gin.Context) {
// HTTP 响应中携带 Cookie
// Set cookie {"label": "ok" }, maxAge 30 seconds.
c.SetCookie("label", "ok", 30, "/", "localhost", false, true)
c.String(200, "Login success!")
})
route.GET("/home", func(c *gin.Context) {
// 获取 name = label 的 Cookie 的 value
if cookie, err := c.Cookie("label"); err == nil {
// 判断 Cookie的value 是否满足预期
if cookie == "ok" {
c.JSON(200, gin.H{"data": "Your home page"})
}
} else {
c.JSON(http.StatusForbidden, gin.H{"error": "Forbidden with no cookie"})
}
})
route.Run(":8080")
}
首先是一个 /login 请求路由,通过 SetCookie 方法给客户端返回 Cookie ,基于此返回用户 Token .
然后 /home 路由的处理,则是通过 gin.Context 中 Cookie 方法获取到HTTP请求头部中 Cookie 的信息 ,然后验证 Cookie 中的value是否满足预期.
这个是一个简单的代码示例,比如用户身份认证机制等,则需要自行完善,这里不再完整展示.
下面将简单描述 gin.Context 对象中 SetCookie 方法和 Cookie 方法的实现原理,帮助读者更好使用这两个 API .
SetCookie 方法的实现原理如下,首先, SetCookie 方法会创建一个 http.Cookie 对象,并设置其名称、值、路径、域名、过期时间等属性。例如,以下代码创建了一个名为 sessionid 的 Cookie
cookie := &http.Cookie{
Name: "sessionid",
Value: "1234",
Expires: time.Now().Add(24 * time.Hour),
Path: "/",
Domain: "",
Secure: false,
HttpOnly:true,
}
接下来,将上述 Cookie 对象转换为字符串格式,并设置到HTTP响应头的 Set-Cookie 字段中。代码实现如下:
func SetCookie(w ResponseWriter, cookie *Cookie) {
if v := cookie.String(); v != "" {
w.Header().Add("Set-Cookie", v)
}
}
这里第三行将 Cookie 存储到 Header 对象当中, Header 是专门用于存储HTTP响应头部的信息。调用 Add 方法时,会根据指定的 Key ,在 Header 对象中查找相应的值列表。如果这个键不存在,则会在 Header 对象中创建一个新的值列表;否则,会在已有的值列表末尾添加新的值,大概流程如下
在返回HTTP响应时,会遍历 Header 对象,填充HTTP响应头部信息,然后返回给客户端,比如上面 Header 生成的HTTP响应头部如下
Set-Cookie: v1
Set-Cookie: v2
Agent: Windows
SetCookie 方法便是通过上述所说流程,将 Cookie 的信息设置到HTTP响应体头部当中去,然后返回给客户端.
在调用 Cookie() 方法时,系统会首先检查请求头部中是否包含名为 Cookie 的字段。如果该字段不存在,则返回空字符串.
如果请求头部中包含 Cookie 字段,同时 Cookie 的 name 为调用 Cookie() 方法指定的值,则系统会解析该字段并将其转换为一个 http.Cookie 对象。这个对象包含了所有的 Cookie 属性,例如名称、值、路径、过期时间、域名等等。最后,返回转换后的 http.Cookie 对象中值,大概流程如下
总的来说, Cookie() 方法的实现原理比较简单,它只是通过查找 HTTP 请求头部中的 Cookie 信息,并将其转换为 http.Cookie 对象来获取请求中特定 Cookie 值.
在本文中,我们深入探讨了Web应用程序在处理用户信息时所面临的挑战,特别是在HTTP协议作为无状态协议的背景下。我们从这一矛盾出发,介绍了解决方案中的 Token 机制,并引出了基于 Cookie 的实现方案.
我们详细阐述了 Cookie 的规范,包括服务端如何发送 Cookie 以及客户端如何在请求中携带 Cookie 信息.
我们进一步深入探讨了具体的实现方式,并介绍了 Gin 框架提供的 API ,这些 API 使得在服务端按照 Cookie 规范发送和解析Cookie变得更加容易。通过一个实际的代码示例,我们演示了如何使用这些API来构建一个基于Cookie实现用户身份验证的Web应用程序.
在探讨 API 的使用之余,我们也深入剖析了Gin框架提供的API的实现原理,为读者提供了更深层次的理解.
基于此,完成了对Gin中Cookie支持的介绍,希望对你有所帮助.
最后此篇关于一文了解Gin对Cookie的支持的文章就讲到这里了,如果你想了解更多关于一文了解Gin对Cookie的支持的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我开始在 Ethereum blockchain 上了解如何开发智能合约以及如何写 web-script用于与智能合约交互(购买、销售、统计......)我得出了该怎么做的结论。我想知道我是否正确理解
我正在 UIView 中使用 CATransform3DMakeRotation,并且我正在尝试进行 45º,变换就像向后放置一样: 这是我拥有的“代码”,但显然没有这样做。 CATransform3
我目前正在测试 WebRTC 的功能,但我有一些脑逻辑问题。 WebRTC 究竟是什么? 我只读了“STUN”、“P2P”和其他...但是在技术方面什么是正确的 WebRTC(见下一个) 我需要什么
我在看 DelayedInit在 Scala in Depth ... 注释是我对代码的理解。 下面的 trait 接受一个非严格计算的参数(由于 => ),并返回 Unit .它的行为类似于构造函数
谁能给我指出一个用图片和简单的代码片段解释 WCF 的资源。我厌倦了谷歌搜索并在所有搜索结果中找到相同的“ABC”文章。 最佳答案 WCF 是一项非常复杂的技术,在我看来,它的文档记录非常少。启动和运
我期待以下 GetArgs.hs打印出传递给它的参数。 import System.Environment main = do args main 3 4 3 :39:1: Coul
private int vbo; private int ibo; vbo = glGenBuffers(); ibo = glGenBuffers(); glBindBuffer(GL_ARRAY_
我正在尝试一个 for 循环。我添加了一个 if 语句以在循环达到 30 时停止循环。 我见过i <= 10将运行 11 次,因为循环在达到 10 次时仍会运行。 如果有设置 i 的 if 语句,为什
我正在尝试了解 WSGI 的功能并需要一些帮助。 到目前为止,我知道它是一种服务器和应用程序之间的中间件,用于将不同的应用程序框架(位于服务器端)与应用程序连接,前提是相关框架具有 WSGI 适配器。
我是 Javascript 的新手,我正在尝试绕过 while 循环。我了解它们的目的,我想我了解它们的工作原理,但我在使用它们时遇到了麻烦。 我希望 while 值自身重复,直到两个随机数相互匹配。
我刚刚偶然发现Fabric并且文档并没有真正说明它是如何工作的。 我有根据的猜测是您需要在客户端和服务器端都安装它。 Python 代码存储在客户端,并在命令运行时通过 Fabric 的有线协议(pr
我想了解 ConditionalWeakTable .和有什么区别 class ClassA { static readonly ConditionalWeakTable OtherClass
关闭。这个问题需要更多focused .它目前不接受答案。 想改善这个问题吗?更新问题,使其仅关注一个问题 editing this post . 5年前关闭。 Improve this questi
我还没有成功找到任何可以引导我理解 UIPickerView 和 UIPickerView 模型的好例子。有什么建议吗? 最佳答案 为什么不使用默认的 Apple 文档示例?这是来自苹果文档的名为 U
我在看foldM为了获得关于如何使用它的直觉。 foldM :: Monad m => (a -> b -> m a) -> a -> [b] -> m a 在这个简单的例子中,我只返回 [Just
答案What are _mm_prefetch() locality hints?详细说明提示的含义。 我的问题是:我想要哪一个? 我正在处理一个被重复调用数十亿次的函数,其中包含一些 int 参数。
我一直在读这个article了解 gcroot 模板。我明白 gcroot provides handles into the garbage collected heap 然后 the handle
提供了一个用例: 流处理架构;事件进入 Kafka,然后由带有 MongoDB 接收器的作业进行处理。 数据库名称:myWebsite集合:用户 并且作业接收 users 集合中的 user 记录。
你好 我想更详细地了解 NFS 文件系统。我偶然发现了《NFS 图解》这本书,不幸的是它只能作为谷歌图书提供,所以有些页面丢失了。有人可能有另一个很好的资源,这将是在较低级别上了解 NFS 的良好开始
我无法理解这个问题,哪个更随机? rand() 或: rand() * rand() 我发现这是一个真正的脑筋急转弯,你能帮我吗? 编辑: 凭直觉,我知道数学答案是它们同样随机,但我忍不住认为,如果您
我是一名优秀的程序员,十分优秀!