只有光头才能变强。html
文本已收录至个人GitHub仓库,欢迎Star:https://github.com/ZhongFuCheng3y/3yjava
在我实习以前我就已经在看单点登陆的是什么了,可是实习的时候一直在忙其余的事,因此有几个网站就一直躺在个人收藏夹里边:git
在前阵子有个读者来我这投稿,是使用JWT实现单点登陆的(可是文章中并无介绍什么是单点登陆),因此我以为是时候来整理一下了。github
单点登陆的英文名叫作:Single Sign On(简称SSO)。web
在初学/之前的时候,通常咱们就单系统,全部的功能都在同一个系统上。redis
后来,咱们为了合理利用资源和下降耦合性,因而把单系统拆分成多个子系统。数据库
好比阿里系的淘宝和天猫,很明显地咱们能够知道这是两个系统,可是你在使用的时候,登陆了天猫,淘宝也会自动登陆。json
简单来讲,单点登陆就是在多个系统中,用户只需一次登陆,各个系统便可感知该用户已经登陆。跨域
在我初学JavaWeb的时候,登陆和注册是我作得最多的一个功能了(初学Servlet的时候作过、学SpringMVC的时候作过、跟着作项目的时候作过…),反正我也数不清我作了多少次登陆和注册的功能了...这里简单讲述一下咱们初学时是怎么作登陆功能的。浏览器
众所周知,HTTP是无状态的协议,这意味着服务器没法确认用户的信息。因而乎,W3C就提出了:给每个用户都发一个通行证,不管谁访问的时候都须要携带通行证,这样服务器就能够从通行证上确认用户的信息。通行证就是Cookie。
若是说Cookie是检查用户身上的”通行证“来确认用户的身份,那么Session就是经过检查服务器上的”客户明细表“来确认用户的身份的。Session至关于在服务器中创建了一份“客户明细表”。
HTTP协议是无状态的,Session不能依据HTTP链接来判断是否为同一个用户。因而乎:服务器向用户浏览器发送了一个名为JESSIONID的Cookie,它的值是Session的id值。其实Session是依据Cookie来识别是不是同一个用户。
因此,通常咱们单系统实现登陆会这样作:
我以前Demo的代码,能够参考一下:
/** * 用户登录 */ @PostMapping(value = "/user/session", produces = {"application/json;charset=UTF-8"}) public Result login(String mobileNo, String password, String inputCaptcha, HttpSession session, HttpServletResponse response) { //判断验证码是否正确 if (WebUtils.validateCaptcha(inputCaptcha, "captcha", session)) { //判断有没有该用户 User user = userService.userLogin(mobileNo, password); if (user != null) { /*设置自动登录,一个星期. 将token保存在数据库中*/ String loginToken = WebUtils.md5(new Date().toString() + session.getId()); user.setLoginToken(loginToken); User user1 = userService.userUpload(user); session.setAttribute("user", user1); CookieUtil.addCookie(response,"loginToken",loginToken,604800); return ResultUtil.success(user1); } else { return ResultUtil.error(ResultEnum.LOGIN_ERROR); } } else { return ResultUtil.error(ResultEnum.CAPTCHA_ERROR); } } /** * 用户退出 */ @DeleteMapping(value = "/session", produces = {"application/json;charset=UTF-8"}) public Result logout(HttpSession session,HttpServletRequest request,HttpServletResponse response ) { //删除session和cookie session.removeAttribute("user"); CookieUtil.clearCookie(request, response, "loginToken"); return ResultUtil.success(); } /** * @author ozc * @version 1.0 * <p> * 拦截器;实现自动登录功能 */ public class UserInterceptor implements HandlerInterceptor { @Autowired private UserService userService; public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object o) throws Exception { User sessionUser = (User) request.getSession().getAttribute("user"); // 已经登录了,放行 if (sessionUser != null) { return true; } else { //获得带过来cookie是否存在 String loginToken = CookieUtil.findCookieByName(request, "loginToken"); if (StringUtils.isNotBlank(loginToken)) { //到数据库查询有没有该Cookie User user = userService.findUserByLoginToken(loginToken); if (user != null) { request.getSession().setAttribute("user", user); return true; } else { //没有该Cookie与之对应的用户(Cookie不匹配) CookieUtil.clearCookie(request, response, "loginToken"); return false; } } else { //没有cookie、也没有登录。是index请求获取用户信息,能够放行 if (request.getRequestURI().contains("session")) { return true; } //没有cookie凭证 response.sendRedirect("/login.html"); return false; } } } }
总结一下上面代码的思路:
若是没看懂的同窗,建议回顾Session和Cookie和HTTP:
单系统登陆功能主要是用Session保存用户信息来实现的,但咱们清楚的是:多系统便可能有多个Tomcat,而Session是依赖当前系统的Tomcat,因此系统A的Session和系统B的Session是不共享的。
解决系统之间Session不共享问题有一下几种方案:
咱们能够将登陆功能单独抽取出来,作成一个子系统。
SSO(登陆系统)的逻辑以下:
// 登陆功能(SSO单独的服务) @Override public TaotaoResult login(String username, String password) throws Exception { //根据用户名查询用户信息 TbUserExample example = new TbUserExample(); Criteria criteria = example.createCriteria(); criteria.andUsernameEqualTo(username); List<TbUser> list = userMapper.selectByExample(example); if (null == list || list.isEmpty()) { return TaotaoResult.build(400, "用户不存在"); } //核对密码 TbUser user = list.get(0); if (!DigestUtils.md5DigestAsHex(password.getBytes()).equals(user.getPassword())) { return TaotaoResult.build(400, "密码错误"); } //登陆成功,把用户信息写入redis //生成一个用户token String token = UUID.randomUUID().toString(); jedisCluster.set(USER_TOKEN_KEY + ":" + token, JsonUtils.objectToJson(user)); //设置session过时时间 jedisCluster.expire(USER_TOKEN_KEY + ":" + token, SESSION_EXPIRE_TIME); return TaotaoResult.ok(token); }
其余子系统登陆时,请求SSO(登陆系统)进行登陆,将返回的token写到Cookie中,下次访问时则把Cookie带上:
public TaotaoResult login(String username, String password, HttpServletRequest request, HttpServletResponse response) { //请求参数 Map<String, String> param = new HashMap<>(); param.put("username", username); param.put("password", password); //登陆处理 String stringResult = HttpClientUtil.doPost(REGISTER_USER_URL + USER_LOGIN_URL, param); TaotaoResult result = TaotaoResult.format(stringResult); //登陆出错 if (result.getStatus() != 200) { return result; } //登陆成功后把取token信息,并写入cookie String token = (String) result.getData(); //写入cookie CookieUtils.setCookie(request, response, "TT_TOKEN", token); //返回成功 return result; }
总结:
到这里,其实咱们会发现其实就两个变化:
上面咱们解决了Session不能共享的问题,但其实还有另外一个问题。Cookie是不能跨域的
好比说,咱们请求<https://www.google.com/>
时,浏览器会自动把google.com
的Cookie带过去给google
的服务器,而不会把<https://www.baidu.com/>
的Cookie带过去给google
的服务器。
这就意味着,因为域名不一样,用户向系统A登陆后,系统A返回给浏览器的Cookie,用户再请求系统B的时候不会将系统A的Cookie带过去。
针对Cookie存在跨域问题,有几种解决方案:
到这里,咱们已经能够实现单点登陆了。
说到单点登陆,就确定会见到这个名词:CAS (Central Authentication Service),下面说说CAS是怎么搞的。
若是已经将登陆单独抽取成系统出来,咱们还能这样玩。如今咱们有两个系统,分别是www.java3y.com
和www.java4y.com
,一个SSOwww.sso.com
首先,用户想要访问系统Awww.java3y.com
受限的资源(好比说购物车功能,购物车功能须要登陆后才能访问),系统Awww.java3y.com
发现用户并无登陆,因而重定向到sso认证中心,并将本身的地址做为参数。请求的地址以下:
www.sso.com?service=www.java3y.com
sso认证中心发现用户未登陆,将用户引导至登陆页面,用户进行输入用户名和密码进行登陆,用户与认证中心创建全局会话(生成一份Token,写到Cookie中,保存在浏览器上)
随后,认证中心重定向回系统A,并把Token携带过去给系统A,重定向的地址以下:
www.java3y.com?token=xxxxxxx
接着,系统A去sso认证中心验证这个Token是否正确,若是正确,则系统A和用户创建局部会话(建立Session)。到此,系统A和用户已是登陆状态了。
此时,用户想要访问系统Bwww.java4y.com
受限的资源(好比说订单功能,订单功能须要登陆后才能访问),系统Bwww.java4y.com
发现用户并无登陆,因而重定向到sso认证中心,并将本身的地址做为参数。请求的地址以下:
www.sso.com?service=www.java4y.com
注意,由于以前用户与认证中心www.sso.com
已经创建了全局会话(当时已经把Cookie保存到浏览器上了),因此此次系统B重定向到认证中心www.sso.com
是能够带上Cookie的。
认证中心根据带过来的Cookie发现已经与用户创建了全局会话了,认证中心重定向回系统B,并把Token携带过去给系统B,重定向的地址以下:
www.java4y.com?token=xxxxxxx
接着,系统B去sso认证中心验证这个Token是否正确,若是正确,则系统B和用户创建局部会话(建立Session)。到此,系统B和用户已是登陆状态了。
看到这里,其实SSO认证中心就相似一个中转站。
参考资料:
乐于输出干货的Java技术公众号:Java3y。公众号内有200多篇原创技术文章、海量视频资源、精美脑图,关注便可获取!
以为个人文章写得不错,点赞!