Skip to content

Commit ea4b53d

Browse files
committed
Add auto-configuration for an indexed reactive session repository
Closes gh-42604
1 parent fc091f7 commit ea4b53d

File tree

3 files changed

+159
-20
lines changed

3 files changed

+159
-20
lines changed

spring-boot-project/spring-boot-autoconfigure/src/dockerTest/java/org/springframework/boot/autoconfigure/session/ReactiveSessionAutoConfigurationRedisTests.java

Lines changed: 94 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -18,28 +18,38 @@
1818

1919
import java.time.Duration;
2020
import java.util.List;
21+
import java.util.Map;
2122

2223
import com.redis.testcontainers.RedisContainer;
2324
import org.junit.jupiter.api.Test;
2425
import org.testcontainers.junit.jupiter.Container;
2526
import org.testcontainers.junit.jupiter.Testcontainers;
27+
import reactor.core.publisher.Mono;
2628

2729
import org.springframework.boot.autoconfigure.AutoConfigurations;
2830
import org.springframework.boot.autoconfigure.data.redis.RedisAutoConfiguration;
2931
import org.springframework.boot.autoconfigure.data.redis.RedisReactiveAutoConfiguration;
32+
import org.springframework.boot.autoconfigure.web.ServerProperties;
3033
import org.springframework.boot.autoconfigure.web.reactive.WebSessionIdResolverAutoConfiguration;
3134
import org.springframework.boot.test.context.FilteredClassLoader;
3235
import org.springframework.boot.test.context.assertj.AssertableReactiveWebApplicationContext;
3336
import org.springframework.boot.test.context.runner.ContextConsumer;
3437
import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner;
3538
import org.springframework.boot.testsupport.container.TestImage;
39+
import org.springframework.data.redis.connection.ReactiveRedisConnection;
40+
import org.springframework.data.redis.connection.ReactiveRedisConnectionFactory;
41+
import org.springframework.data.redis.connection.RedisConnectionFactory;
3642
import org.springframework.http.ResponseCookie;
3743
import org.springframework.session.MapSession;
3844
import org.springframework.session.SaveMode;
3945
import org.springframework.session.data.mongo.ReactiveMongoSessionRepository;
46+
import org.springframework.session.data.redis.ReactiveRedisIndexedSessionRepository;
4047
import org.springframework.session.data.redis.ReactiveRedisSessionRepository;
48+
import org.springframework.session.data.redis.config.ConfigureReactiveRedisAction;
49+
import org.springframework.session.data.redis.config.annotation.ConfigureNotifyKeyspaceEventsReactiveAction;
4150

4251
import static org.assertj.core.api.Assertions.assertThat;
52+
import static org.assertj.core.api.Assertions.entry;
4353

4454
/**
4555
* Reactive Redis-specific tests for {@link SessionAutoConfiguration}.
@@ -121,6 +131,52 @@ void sessionCookieConfigurationIsAppliedToAutoConfiguredWebSessionIdResolver() {
121131
}));
122132
}
123133

134+
@Test
135+
void indexedRedisSessionDefaultConfig() {
136+
this.contextRunner
137+
.withPropertyValues("spring.session.redis.repository-type=indexed",
138+
"spring.data.redis.host=" + redis.getHost(), "spring.data.redis.port=" + redis.getFirstMappedPort())
139+
.withConfiguration(AutoConfigurations.of(RedisAutoConfiguration.class))
140+
.run(validateSpringSessionUsesIndexedRedis("spring:session:", SaveMode.ON_SET_ATTRIBUTE));
141+
}
142+
143+
@Test
144+
void indexedRedisSessionStoreWithCustomizations() {
145+
this.contextRunner.withConfiguration(AutoConfigurations.of(RedisAutoConfiguration.class))
146+
.withPropertyValues("spring.session.redis.repository-type=indexed", "spring.session.redis.namespace=foo",
147+
"spring.session.redis.save-mode=on-get-attribute", "spring.data.redis.host=" + redis.getHost(),
148+
"spring.data.redis.port=" + redis.getFirstMappedPort())
149+
.run(validateSpringSessionUsesIndexedRedis("foo:", SaveMode.ON_GET_ATTRIBUTE));
150+
}
151+
152+
@Test
153+
void indexedRedisSessionWithConfigureActionNone() {
154+
this.contextRunner.withConfiguration(AutoConfigurations.of(RedisAutoConfiguration.class))
155+
.withPropertyValues("spring.session.redis.repository-type=indexed",
156+
"spring.session.redis.configure-action=none", "spring.data.redis.host=" + redis.getHost(),
157+
"spring.data.redis.port=" + redis.getFirstMappedPort())
158+
.run(validateStrategy(ConfigureReactiveRedisAction.NO_OP.getClass()));
159+
}
160+
161+
@Test
162+
void indexedRedisSessionWithDefaultConfigureActionNone() {
163+
this.contextRunner.withConfiguration(AutoConfigurations.of(RedisAutoConfiguration.class))
164+
.withPropertyValues("spring.session.redis.repository-type=indexed",
165+
"spring.data.redis.host=" + redis.getHost(), "spring.data.redis.port=" + redis.getFirstMappedPort())
166+
.run(validateStrategy(ConfigureNotifyKeyspaceEventsReactiveAction.class,
167+
entry("notify-keyspace-events", "gxE")));
168+
}
169+
170+
@Test
171+
void indexedRedisSessionWithCustomConfigureReactiveRedisActionBean() {
172+
this.contextRunner.withConfiguration(AutoConfigurations.of(RedisAutoConfiguration.class))
173+
.withUserConfiguration(MaxEntriesReactiveRedisAction.class)
174+
.withPropertyValues("spring.session.redis.repository-type=indexed",
175+
"spring.data.redis.host=" + redis.getHost(), "spring.data.redis.port=" + redis.getFirstMappedPort())
176+
.run(validateStrategy(MaxEntriesReactiveRedisAction.class, entry("set-max-intset-entries", "1024")));
177+
178+
}
179+
124180
private ContextConsumer<AssertableReactiveWebApplicationContext> validateSpringSessionUsesRedis(String namespace,
125181
SaveMode saveMode) {
126182
return (context) -> {
@@ -133,4 +189,42 @@ private ContextConsumer<AssertableReactiveWebApplicationContext> validateSpringS
133189
};
134190
}
135191

192+
private ContextConsumer<AssertableReactiveWebApplicationContext> validateSpringSessionUsesIndexedRedis(
193+
String keyNamespace, SaveMode saveMode) {
194+
return (context) -> {
195+
ReactiveRedisIndexedSessionRepository repository = validateSessionRepository(context,
196+
ReactiveRedisIndexedSessionRepository.class);
197+
assertThat(repository).hasFieldOrPropertyWithValue("defaultMaxInactiveInterval",
198+
new ServerProperties().getReactive().getSession().getTimeout());
199+
assertThat(repository).hasFieldOrPropertyWithValue("namespace", keyNamespace);
200+
assertThat(repository).hasFieldOrPropertyWithValue("saveMode", saveMode);
201+
};
202+
}
203+
204+
private ContextConsumer<AssertableReactiveWebApplicationContext> validateStrategy(
205+
Class<? extends ConfigureReactiveRedisAction> expectedConfigureReactiveRedisActionType,
206+
Map.Entry<?, ?>... expectedConfig) {
207+
return (context) -> {
208+
assertThat(context).hasSingleBean(ConfigureReactiveRedisAction.class);
209+
assertThat(context).hasSingleBean(RedisConnectionFactory.class);
210+
assertThat(context.getBean(ConfigureReactiveRedisAction.class))
211+
.isInstanceOf(expectedConfigureReactiveRedisActionType);
212+
ReactiveRedisConnection connection = context.getBean(ReactiveRedisConnectionFactory.class)
213+
.getReactiveConnection();
214+
if (expectedConfig.length > 0) {
215+
assertThat(connection.serverCommands().getConfig("*").block(Duration.ofSeconds(30)))
216+
.contains(expectedConfig);
217+
}
218+
};
219+
}
220+
221+
static class MaxEntriesReactiveRedisAction implements ConfigureReactiveRedisAction {
222+
223+
@Override
224+
public Mono<Void> configure(ReactiveRedisConnection connection) {
225+
return Mono.when(connection.serverCommands().setConfig("set-max-intset-entries", "1024"));
226+
}
227+
228+
}
229+
136230
}

spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisReactiveSessionConfiguration.java

Lines changed: 56 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2012-2023 the original author or authors.
2+
* Copyright 2012-2024 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -19,6 +19,7 @@
1919
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
2020
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
2121
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
22+
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
2223
import org.springframework.boot.autoconfigure.web.ServerProperties;
2324
import org.springframework.boot.context.properties.EnableConfigurationProperties;
2425
import org.springframework.boot.context.properties.PropertyMapper;
@@ -28,7 +29,11 @@
2829
import org.springframework.data.redis.connection.ReactiveRedisConnectionFactory;
2930
import org.springframework.session.ReactiveSessionRepository;
3031
import org.springframework.session.config.ReactiveSessionRepositoryCustomizer;
32+
import org.springframework.session.data.redis.ReactiveRedisIndexedSessionRepository;
3133
import org.springframework.session.data.redis.ReactiveRedisSessionRepository;
34+
import org.springframework.session.data.redis.config.ConfigureReactiveRedisAction;
35+
import org.springframework.session.data.redis.config.annotation.ConfigureNotifyKeyspaceEventsReactiveAction;
36+
import org.springframework.session.data.redis.config.annotation.web.server.RedisIndexedWebSessionConfiguration;
3237
import org.springframework.session.data.redis.config.annotation.web.server.RedisWebSessionConfiguration;
3338

3439
/**
@@ -43,20 +48,58 @@
4348
@ConditionalOnMissingBean(ReactiveSessionRepository.class)
4449
@ConditionalOnBean(ReactiveRedisConnectionFactory.class)
4550
@EnableConfigurationProperties(RedisSessionProperties.class)
46-
@Import(RedisWebSessionConfiguration.class)
4751
class RedisReactiveSessionConfiguration {
4852

49-
@Bean
50-
ReactiveSessionRepositoryCustomizer<ReactiveRedisSessionRepository> springBootSessionRepositoryCustomizer(
51-
SessionProperties sessionProperties, RedisSessionProperties redisSessionProperties,
52-
ServerProperties serverProperties) {
53-
return (sessionRepository) -> {
54-
PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull();
55-
map.from(sessionProperties.determineTimeout(() -> serverProperties.getReactive().getSession().getTimeout()))
56-
.to(sessionRepository::setDefaultMaxInactiveInterval);
57-
map.from(redisSessionProperties::getNamespace).to(sessionRepository::setRedisKeyNamespace);
58-
map.from(redisSessionProperties::getSaveMode).to(sessionRepository::setSaveMode);
59-
};
53+
@Configuration(proxyBeanMethods = false)
54+
@ConditionalOnProperty(prefix = "spring.session.redis", name = "repository-type", havingValue = "default",
55+
matchIfMissing = true)
56+
@Import(RedisWebSessionConfiguration.class)
57+
static class DefaultRedisSessionConfiguration {
58+
59+
@Bean
60+
ReactiveSessionRepositoryCustomizer<ReactiveRedisSessionRepository> springBootSessionRepositoryCustomizer(
61+
SessionProperties sessionProperties, RedisSessionProperties redisSessionProperties,
62+
ServerProperties serverProperties) {
63+
return (sessionRepository) -> {
64+
PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull();
65+
map.from(sessionProperties
66+
.determineTimeout(() -> serverProperties.getReactive().getSession().getTimeout()))
67+
.to(sessionRepository::setDefaultMaxInactiveInterval);
68+
map.from(redisSessionProperties::getNamespace).to(sessionRepository::setRedisKeyNamespace);
69+
map.from(redisSessionProperties::getSaveMode).to(sessionRepository::setSaveMode);
70+
};
71+
}
72+
73+
}
74+
75+
@Configuration(proxyBeanMethods = false)
76+
@ConditionalOnProperty(prefix = "spring.session.redis", name = "repository-type", havingValue = "indexed")
77+
@Import(RedisIndexedWebSessionConfiguration.class)
78+
static class IndexedRedisSessionConfiguration {
79+
80+
@Bean
81+
@ConditionalOnMissingBean
82+
ConfigureReactiveRedisAction configureReactiveRedisAction(RedisSessionProperties redisSessionProperties) {
83+
return switch (redisSessionProperties.getConfigureAction()) {
84+
case NOTIFY_KEYSPACE_EVENTS -> new ConfigureNotifyKeyspaceEventsReactiveAction();
85+
case NONE -> ConfigureReactiveRedisAction.NO_OP;
86+
};
87+
}
88+
89+
@Bean
90+
ReactiveSessionRepositoryCustomizer<ReactiveRedisIndexedSessionRepository> springBootSessionRepositoryCustomizer(
91+
SessionProperties sessionProperties, RedisSessionProperties redisSessionProperties,
92+
ServerProperties serverProperties) {
93+
return (sessionRepository) -> {
94+
PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull();
95+
map.from(sessionProperties
96+
.determineTimeout(() -> serverProperties.getReactive().getSession().getTimeout()))
97+
.to(sessionRepository::setDefaultMaxInactiveInterval);
98+
map.from(redisSessionProperties::getNamespace).to(sessionRepository::setRedisKeyNamespace);
99+
map.from(redisSessionProperties::getSaveMode).to(sessionRepository::setSaveMode);
100+
};
101+
}
102+
60103
}
61104

62105
}

spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionProperties.java

Lines changed: 9 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2012-2022 the original author or authors.
2+
* Copyright 2012-2024 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -36,7 +36,7 @@ public class RedisSessionProperties {
3636

3737
/**
3838
* Sessions flush mode. Determines when session changes are written to the session
39-
* store.
39+
* store. Not supported with a reactive session repository.
4040
*/
4141
private FlushMode flushMode = FlushMode.ON_SAVE;
4242

@@ -47,14 +47,15 @@ public class RedisSessionProperties {
4747
private SaveMode saveMode = SaveMode.ON_SET_ATTRIBUTE;
4848

4949
/**
50-
* The configure action to apply when no user defined ConfigureRedisAction bean is
51-
* present.
50+
* The configure action to apply when no user-defined ConfigureRedisAction or
51+
* ConfigureReactiveRedisAction bean is present.
5252
*/
5353
private ConfigureAction configureAction = ConfigureAction.NOTIFY_KEYSPACE_EVENTS;
5454

5555
/**
5656
* Cron expression for expired session cleanup job. Only supported when
57-
* repository-type is set to indexed.
57+
* repository-type is set to indexed. Not supported with a reactive session
58+
* repository.
5859
*/
5960
private String cleanupCron;
6061

@@ -135,12 +136,13 @@ public enum ConfigureAction {
135136
public enum RepositoryType {
136137

137138
/**
138-
* Auto-configure a RedisSessionRepository.
139+
* Auto-configure a RedisSessionRepository or ReactiveRedisSessionRepository.
139140
*/
140141
DEFAULT,
141142

142143
/**
143-
* Auto-configure a RedisIndexedSessionRepository.
144+
* Auto-configure a RedisIndexedSessionRepository or
145+
* ReactiveRedisIndexedSessionRepository.
144146
*/
145147
INDEXED
146148

0 commit comments

Comments
 (0)