SpringBoot2.x版本中,使用SpringSession踩的坑及解決

SpringBoot2.x SpringSession踩坑

Exception encountered during context initialization – cancelling refresh attempt: org.springframework.beans.factory.BeanCreationException: Error creating bean with name ‘org.springframework.boot.autoconfigure.session.SessionAutoConfiguration$ServletSessionRepositoryValidator’: Invocation of init method failed; nested exception is org.springframework.boot.autoconfigure.session.SessionRepositoryUnavailableException: No session repository could be auto-configured, check your configuration (session store type is ‘redis’)

這是因為缺少瞭spring-session-data-redis依賴。

關於SpringBoot2.X中,引用SpringSession,同時使用Redis存儲緩存數據需要進行如下配置:

 <!--SpringSession依賴-->
 <dependency>
  <groupId>org.springframework.session</groupId>
  <artifactId>spring-session-core</artifactId>
 </dependency>
 <!--SpringSessionRedis依賴-->
 <dependency>
  <groupId>org.springframework.session</groupId>
  <artifactId>spring-session-data-redis</artifactId>
 </dependency>
#使用使用Redis緩存session數據
spring.session.store-type=REDIS
#Redis服務器地址
spring.redis.host=127.0.0.1
#Redis服務器端口號
spring.redis.port=6379

總結:

在SpringBoot2.x的版本中,引用spring-session-core時,不是對spring-session-data-redis進行加載,需要用戶自己添加關於spring-session與redis的關聯依賴。

springboot 2.x 踩坑——跨域導致session問題

目前IT界主流前後端分離,但是在分離過程中一定會存在跨域的問題。

什麼是跨域?

是指瀏覽器從一個域名的網頁去請求另一個域名的資源時,域名、端口、協議任一不同,都是跨域。

遇到的場景

當我們用springboot + shrio +vue 來做後臺管理的項目時,無法獲取shiroSession當前登錄的用戶,

於是我們就排查,網上說在跨域時讓session通過就可以瞭

後端

@Configuration
public class CorsConfig {
    private CorsConfiguration buildConfig() {
        CorsConfiguration corsConfiguration = new CorsConfiguration();
        corsConfiguration.setAllowCredentials(true);
        // 允許任何域名使用
        corsConfiguration.addAllowedOrigin("*");
        // 允許任何頭
        corsConfiguration.addAllowedHeader("*");
        // 允許任何方法(post、get等)
        corsConfiguration.addAllowedMethod("*");
        corsConfiguration.setMaxAge(3600L);
        return corsConfiguration;
    }
    @Bean
    public CorsFilter corsFilter() {
        UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
        // 對接口配置跨域設置
        source.registerCorsConfiguration("/**", buildConfig());
        return new CorsFilter(source);
    }
}

前端

axios.defaults.withCredentials=true; 

但是設置後依舊不行

經過一天的百度與排查,我回滾到springboot 1.x 居然沒有這個問題,才定位到是升級到springboot 2.x導致的原因,好瞭,已經抓住兇手瞭,這下子好對癥下藥瞭,去網上看瞭 springboot升級到2.x spring session 相關的問題。

終於發現瞭新大陸,spring-session 2.x 中 Cookie裡面居然引入瞭SameSite 這個叼毛,他默認值是 Lax,好瞭咱們來看看這個是什麼東西?

SameSite Cookie 是用來防止CSRF攻擊,它有兩個值:Strict、Lax

SameSite = Strict:

意為嚴格模式,表明這個cookie在任何情況下都不可能作為第三方cookie;

SameSite = Lax:

意為寬松模式,在GET請求是可以作為第三方cookie,但是不能攜帶cookie進行跨域post訪問(這就很蛋疼瞭,我們那個校驗接口就是POST請求)

總結:前端請求到後臺,每次session都不一樣,每次都是新的會話,導致獲取不到用戶信息

解決方案:

將SameSite設置為空

@Configuration
public class SpringSessionConfig { 
    @Bean
    public CookieSerializer httpSessionIdResolver() {
        DefaultCookieSerializer cookieSerializer = new DefaultCookieSerializer();
        // 取消僅限同一站點設置
        cookieSerializer.setSameSite(null);
        return cookieSerializer;
    }
}

問題解決!!

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: