常用配置
本节包含适用于所有或大多数 Spring Session 模块的常用配置。它包含以下用例的配置示例
更改 Session ID 的生成方式
默认情况下,Spring Session 使用 `UuidSessionIdGenerator`,该生成器又使用 `java.util.UUID` 生成 Session ID。在某些情况下,最好包含其他字符以增加熵,或者您可能希望使用不同的算法来生成 Session ID。要更改此设置,您可以提供自定义的 `SessionIdGenerator` bean。
-
Java
@Bean
public SessionIdGenerator sessionIdGenerator() {
return new MySessionIdGenerator();
}
class MySessionIdGenerator implements SessionIdGenerator {
@Override
public String generate() {
// ...
}
}
公开您的 `SessionIdGenerator` bean 后,Spring Session 将使用它来生成 Session ID。
如果您是手动配置 `SessionRepository` bean(例如,不使用 `@EnableRedisHttpSession`),则可以直接在 `SessionRepository` 实现上设置 `SessionIdGenerator`。
-
Java
@Bean
public RedisSessionRepository redisSessionRepository(RedisOperations redisOperations) {
RedisSessionRepository repository = new RedisSessionRepository(redisOperations)
repository.setSessionIdGenerator(new MySessionIdGenerator());
return repository;
}
自定义 Session Cookie
设置 Spring Session 后,您可以通过公开 `CookieSerializer` 作为 Spring bean 来自定义会话 Cookie 的写入方式。Spring Session 自带 `DefaultCookieSerializer`。将 `DefaultCookieSerializer` 作为 Spring bean 公开会在您使用 `@EnableRedisHttpSession` 等配置时增强现有配置。以下示例显示如何自定义 Spring Session 的 Cookie。
@Bean
public CookieSerializer cookieSerializer() {
DefaultCookieSerializer serializer = new DefaultCookieSerializer();
serializer.setCookieName("JSESSIONID"); (1)
serializer.setCookiePath("/"); (2)
serializer.setDomainNamePattern("^.+?\\.(\\w+\\.[a-z]+)$"); (3)
return serializer;
}
1 | 我们将 Cookie 的名称自定义为 `JSESSIONID`。 |
2 | 我们将 Cookie 的路径自定义为 `/`(而不是上下文根的默认值)。 |
3 | 我们将域名模式(正则表达式)自定义为 `^.?\\.(\\w\\.[a-z]+)$`。这允许跨域和应用程序共享会话。如果正则表达式不匹配,则不设置任何域,并使用现有域。如果正则表达式匹配,则第一个分组将用作域。这意味着对child.example.com 的请求将域设置为 `example.com`。但是,对localhost:8080/ 或192.168.1.100:8080/ 的请求将使 Cookie 保持未设置状态,因此在开发中仍然有效,无需对生产环境进行任何更改。 |
您应该只匹配有效的域名字符,因为域名会反映在响应中。这样做可以防止恶意用户执行诸如HTTP 响应拆分之类的攻击。 |
配置选项
提供以下配置选项:
-
cookieName
:要使用的 Cookie 名称。默认值:SESSION
。 -
useSecureCookie
:指定是否应使用安全 Cookie。默认值:在创建时使用 `HttpServletRequest.isSecure()` 的值。 -
cookiePath
:Cookie 的路径。默认值:上下文根。 -
cookieMaxAge
:指定在创建会话时要设置的 Cookie 的最大生存期。默认值:-1
,表示浏览器关闭时应删除 Cookie。 -
jvmRoute
:指定要附加到 Session ID 并包含在 Cookie 中的后缀。用于标识要为此会话关联路由到的 JVM。对于某些实现(例如 Redis),此选项不会带来性能优势。但是,它可以帮助跟踪特定用户的日志。 -
domainName
:允许指定要用于 Cookie 的特定域名。此选项易于理解,但在开发和生产环境之间通常需要不同的配置。请参阅 `domainNamePattern` 作为替代方案。 -
domainNamePattern
:用于从 `HttpServletRequest#getServerName()` 中提取域名的不区分大小写的模式。该模式应提供一个用于提取 Cookie 域值的组。如果正则表达式不匹配,则不设置任何域,并使用现有域。如果正则表达式匹配,则第一个分组将用作域。 -
sameSite
:`SameSite` Cookie 指令的值。要禁用 `SameSite` Cookie 指令的序列化,您可以将此值设置为 `null`。默认值:`Lax` -
rememberMeRequestAttribute
:指示记住我登录的请求属性名称。如果指定,则 Cookie 将写入为 `Integer.MAX_VALUE`。
如果您使用的是 `SpringSessionRememberMeServices` 并声明了一个自定义的 `DefaultCookieSerializer` bean,则应设置 `rememberMeRequestAttribute` 字段以确保 Spring Session 依赖于会话过期而不是 Cookie 过期。为此,您可以使用以下代码片段:`defaultCookieSerializer.setRememberMeRequestAttribute(SpringSessionRememberMeServices.REMEMBER_ME_LOGIN_ATTR);` |
WebFlux 中的自定义 Cookie
您可以通过公开 `WebSessionIdResolver` 作为 Spring bean 来自定义在 WebFlux 应用程序中会话 Cookie 的写入方式。Spring Session 默认使用 `CookieWebSessionIdResolver`。以下示例显示如何自定义 Spring Session 的 Cookie。
@Bean
public WebSessionIdResolver webSessionIdResolver() {
CookieWebSessionIdResolver resolver = new CookieWebSessionIdResolver();
resolver.setCookieName("JSESSIONID"); (1)
resolver.addCookieInitializer((builder) -> builder.path("/")); (2)
resolver.addCookieInitializer((builder) -> builder.sameSite("Strict")); (3)
return resolver;
}
1 | 我们将 Cookie 的名称自定义为 `JSESSIONID`。 |
2 | 我们将 Cookie 的路径自定义为 `/`(而不是上下文根的默认值)。 |
3 | 我们将 `SameSite` Cookie 指令自定义为 `Strict`。 |
提供 `ReactiveSessionRegistry` 的 Spring Session 实现
Spring Session 提供与 Spring Security 的集成以支持其响应式并发会话控制。这允许限制单个用户可以同时拥有的活动会话数,但是,与默认的 Spring Security 支持不同,这在集群环境中也能正常工作。这是通过提供 Spring Security 的 `ReactiveSessionRegistry` 接口的 `SpringSessionBackedReactiveSessionRegistry` 实现来完成的。
-
Java
@Bean
public <S extends Session> SpringSessionBackedReactiveSessionRegistry<S> sessionRegistry(
ReactiveSessionRepository<S> sessionRepository,
ReactiveFindByIndexNameSessionRepository<S> indexedSessionRepository) {
return new SpringSessionBackedReactiveSessionRegistry<>(sessionRepository, indexedSessionRepository);
}
更多关于如何使用ReactiveSessionRegistry
的方法,请参考Spring Security并发会话控制文档。您也可以查看示例应用程序此处。