有偿问答
面经分享
技术探讨
资料领取
登录
厉害了!单点登录系统用 8 张漫画就解释了。。。
社长
1年前
⋅ 429 阅读
> 本文由 [简悦 SimpRead](http://ksria.com/simpread/) 转码, 原文地址 [blog.leapoahead.com](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/) > 上次在《JSON Web Token - 在 Web 应用间安全地传递信息》中我提到了 JSON Web Token 可以用来设计单点登录系统。 上次在[《JSON Web Token - 在 Web 应用间安全地传递信息》](http://blog.leapoahead.com/2015/09/06/understanding-jwt/)中我提到了 JSON Web Token 可以用来设计单点登录系统。我尝试用八幅漫画先让大家理解如何设计正常的用户认证系统,然后再延伸到单点登录系统。 如果还没有阅读[《JSON Web Token - 在 Web 应用间安全地传递信息》](http://blog.leapoahead.com/2015/09/06/understanding-jwt/),我强烈建议你花十分钟阅读它,理解 JWT 的生成过程和原理。 ### [](#用户认证八步走 "用户认证八步走")用户认证八步走 所谓用户认证(Authentication),就是让用户登录,并且在接下来的一段时间内让用户访问网站时可以使用其账户,而不需要再次登录的机制。 > 小知识:可别把用户认证和用户授权(Authorization)搞混了。用户授权指的是规定并允许用户使用自己的权限,例如发布帖子、管理站点等。 首先,服务器应用(下面简称 “应用”)让用户通过 Web 表单将自己的用户名和密码发送到服务器的接口。这一过程一般是一个 HTTP POST 请求。建议的方式是通过 SSL 加密的传输(https 协议),从而避免敏感信息被嗅探。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth1.png) 接下来,应用和数据库核对用户名和密码。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth2.png) 核对用户名和密码成功后,应用将用户的`id`(图中的`user_id`)作为 JWT Payload 的一个属性,将其与头部分别进行 Base64 编码拼接后签名,形成一个 JWT。这里的 JWT 就是一个形同`lll.zzz.xxx`的字符串。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth3.png) 应用将 JWT 字符串作为该请求 Cookie 的一部分返回给用户。注意,在这里必须使用`HttpOnly`属性来防止 Cookie 被 JavaScript 读取,从而避免[跨站脚本攻击(XSS 攻击)](http://www.cnblogs.com/bangerlee/archive/2013/04/06/3002142.html)。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth4.png) 在 Cookie 失效或者被删除前,用户每次访问应用,应用都会接受到含有`jwt`的 Cookie。从而应用就可以将 JWT 从请求中提取出来。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth5.png) 应用通过一系列任务检查 JWT 的有效性。例如,检查签名是否正确;检查 Token 是否过期;检查 Token 的接收方是否是自己(可选)。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth6.png) 应用在确认 JWT 有效之后,JWT 进行 Base64 解码(可能在上一步中已经完成),然后在 Payload 中读取用户的 id 值,也就是`user_id`属性。这里用户的`id`为 1025。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth7.png) 应用从数据库取到`id`为 1025 的用户的信息,加载到内存中,进行 ORM 之类的一系列底层逻辑初始化。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth8.png) 应用根据用户请求进行响应。 ![](http://blog.leapoahead.com/2015/09/07/user-authentication-with-jwt/jwtauth9.png) ### [](#和Session方式存储id的差异 "和Session方式存储id的差异")和 Session 方式存储 id 的差异 Session 方式存储用户 id 的最大弊病在于要占用大量服务器内存,对于较大型应用而言可能还要保存许多的状态。一般而言,大型应用还需要借助一些 KV 数据库和一系列缓存机制来实现 Session 的存储。 而 JWT 方式将用户状态分散到了客户端中,可以明显减轻服务端的内存压力。除了用户 id 之外,还可以存储其他的和用户相关的信息,例如该用户是否是管理员、用户所在的分桶(见 [《你所应该知道的 A/B 测试基础》一文](/2015/08/27/introduction-to-ab-testing/)等。 虽说 JWT 方式让服务器有一些计算压力(例如加密、编码和解码),但是这些压力相比磁盘 I/O 而言或许是半斤八两。具体是否采用,需要在不同场景下用数据说话。 ### [](#单点登录 "单点登录")单点登录 Session 方式来存储用户 id,一开始用户的 Session 只会存储在一台服务器上。对于有多个子域名的站点,每个子域名至少会对应一台不同的服务器,例如: * [www.taobao.com](http://www.taobao.com/) * nv.taobao.com * nz.taobao.com * login.taobao.com 所以如果要实现在`login.taobao.com`登录后,在其他的子域名下依然可以取到 Session,这要求我们在多台服务器上同步 Session。 使用 JWT 的方式则没有这个问题的存在,因为用户的状态已经被传送到了客户端。因此,我们只需要将含有 JWT 的 Cookie 的`domain`设置为顶级域名即可,例如 ``` Set-Cookie: jwt=lll.zzz.xxx; HttpOnly; max-age=980000; domain=.taobao.com ``` 注意`domain`必须设置为一个点加顶级域名,即`.taobao.com`。这样,taobao.com 和 *.taobao.com 就都可以接受到这个 Cookie,并获取 JWT 了。 对于 JWT 的两篇文章有相关问题的同学请直接在下面的评论区与我讨论(请勿邮件讨论)。如果你感兴趣,你可以在下方订阅我的半月刊,我将给你推送更多精彩的内容;) ### 让有趣易懂的知识主动找到你 订阅我的 Email 半月刊,让我们共同学习、成长。绝无广告!
阅读全部
全部评论:
0
条
我有话说:
@
发送
-- 目录 --
关注官方公众号:
Java问答社
接收最新有赏问答推送!
最新发布
1.
SpringBoot 接口数据加解密技巧,so easy!
2.
一个依赖搞定 Spring Boot 反爬虫,防止接口盗刷!
3.
Java8 Stream 极大简化了代码,它是如何实现的?
4.
马上大四了,秋招还是春招好?先找工作还是找实习?
5.
万字详解 Linux 常用指令(值得收藏)
6.
4年工作经验,多线程间的5种通信方式都说不出来,你敢信?
最新评论
部署文档没有了,您能提供下吗
部署文档没有了,能提供下吗
我测你的🐎
源码从哪里获取请问
想学
那篇石墨文档 没有权限查看哇