基於SpringBoot+Redis的Session共享與單點登入
title: 基於SpringBoot+Redis的Session共享與單點登入
date: 2019-07-23 02:55:52
categories:
- 架構
author: mrzhou
tags: - SpringBoot
- redis
- session
單點登入
基於SpringBoot+Redis的Session共享與單點登入
前言
使用Redis來實現Session共享,其實網上已經有很多例子了,這是確保在叢集部署中最典型的redis使用場景。在SpringBoot專案中,其實可以一行執行程式碼都不用寫,只需要簡單新增新增依賴和一行註解就可以實現(當然配置資訊還是需要的)。
然後簡單地把該專案部署到不同的tomcat下,比如不同的埠(A、B),但專案訪問路徑是相同的。此時在A中使用set方法,然後在B中使用get方法,就可以發現B中可以獲取A中設定的內容。
但如果就把這樣的一個專案在多個tomcat中的部署說實現了單點登入,那就不對了。
所謂單點登入是指在不同的專案中,只需要任何一個專案登入了,其他專案不需要登入。
同樣是上面的例子,我們把set和get兩個方法分別放到兩個專案(set、get)中,並且以叢集方式把兩個專案都部署到伺服器A和B中,然後分別訪問A伺服器的set和B伺服器的get,你就會發現完全得不到你想要的結果。
同一專案中的set/get
依賴新增就不說了,直接使用最簡單的方式
@SpringBootApplication @EnableRedisHttpSession @RestController public class SessionShareApplication { public static void main(String[] args) { SpringApplication.run(SessionShareApplication.class, args); } @Autowired HttpSession session; @Autowired HttpServletRequest req; @GetMapping("/set") public Object set() { session.setAttribute("state", "state was setted."); Map<String, Object> map = new TreeMap<>(); map.put("msg", session.getAttribute("state")); map.put("serverPort", req.getLocalPort()); return map; } @GetMapping("/get") public Object get() { Map<String, Object> map = new TreeMap<>(); map.put("msg", session.getAttribute("state")); map.put("serverPort", req.getLocalPort()); return map; } }
將該專案打war包,分別部署在tomcatA(埠8080),tomcatB(埠8081),然後通過tomcatA/set 方法設定session,再使用 tomcatB/get 方法即可獲得session的值。但這只是實現了同一專案session的共享。並不是單點登入。
為了驗證,我們不仿將set/get方法拆分為兩個專案。
拆分set/get為兩個專案
- get專案
@SpringBootApplication @EnableRedisHttpSession @RestController public class SetApplication { public static void main(String[] args) { SpringApplication.run(SetApplication.class, args); } @Autowired HttpSession session; @Autowired HttpServletRequest req; @GetMapping("/") public Object set() { session.setAttribute("state", "state was setted."); Map<String, Object> map = new TreeMap<>(); map.put("msg", session.getAttribute("state")); map.put("serverPort", req.getLocalPort()); return map; } }
將該專案打包為set.war
- set專案
@SpringBootApplication
@EnableRedisHttpSession
@RestController
public class GetApplication {
public static void main(String[] args) {
SpringApplication.run(GetApplication.class, args);
}
@Autowired
HttpSession session;
@Autowired
HttpServletRequest req;
@GetMapping("/")
public Object get() {
Map<String, Object> map = new TreeMap<>();
map.put("msg", session.getAttribute("state"));
map.put("serverPort", req.getLocalPort());
return map;
}
}
將該專案打包為get.war
再分別將set.war,get.war部署在tomcatA和tomcatB,再通過 tomcatA/set 設定session內容, 然後通過 tomcatB/get 就發現無法獲得session的值。
問題分析
儘管我們使用的路徑都是一樣的,但其實是兩個專案,與前面的一個專案是完全不同的,問題就在於 session和cookie在預設情況下是與專案路徑相關的,在同一個專案的情況下兩個方法所需要的cookie依賴的專案路徑是相同的,所以獲取session的值就沒有問題,但在後一種情況下,cookie的路徑是分別屬於不同的專案的,所以第二個專案就無法獲得第一個專案中設定的session內容了。
解決方法
解決方法在springboot專案中其實也非常簡單。既然cookie路徑發生了變化,那我們讓它配置為相同的路徑就解決了。
在每個子專案中都新增一個配置類或者直接設定cookie的路徑,如果有域名還可以設定域名的限制,比如 set.xxx.com 與 get.xxx.com 這種情況與我們就需要設定cookie的域名為 xxx.com,以確保無法在哪個專案下都能夠獲取 xxx.com 這個域名下的cookie值。這樣就確保能夠正常獲得共享的session值了。
@Configuration
public class CookieConfig {
@Bean
public static DefaultCookieSerializer defaultCookieSerializer() {
DefaultCookieSerializer serializer = new DefaultCookieSerializer();
serializer.setCookiePath("/");
//serializer.setDomainName("xxx.com"); //如果使用域名訪問,建議對這一句進行設定
return serializer;
}
}
以上才是正直的redis實現單點登入的正確開啟方式