diff --git a/spring-boot-devtools/pom.xml b/spring-boot-devtools/pom.xml
index 6fe474293ef..072af3653cd 100644
--- a/spring-boot-devtools/pom.xml
+++ b/spring-boot-devtools/pom.xml
@@ -54,6 +54,16 @@
spring-security-web
true
+
+ org.springframework.data
+ spring-data-redis
+ true
+
+
+ org.springframework.session
+ spring-session
+ true
+
org.springframework.boot
diff --git a/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/LocalDevToolsAutoConfiguration.java b/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/LocalDevToolsAutoConfiguration.java
index 2c916e6bbdf..e4e156742fc 100644
--- a/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/LocalDevToolsAutoConfiguration.java
+++ b/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/LocalDevToolsAutoConfiguration.java
@@ -22,6 +22,7 @@ import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
@@ -40,12 +41,15 @@ import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.event.ContextRefreshedEvent;
import org.springframework.context.event.EventListener;
+import org.springframework.data.redis.core.RedisTemplate;
+import org.springframework.session.ExpiringSession;
import org.springframework.util.StringUtils;
/**
* {@link EnableAutoConfiguration Auto-configuration} for local development support.
*
* @author Phillip Webb
+ * @author Andy Wilkinson
* @since 1.3.0
*/
@Configuration
@@ -159,6 +163,19 @@ public class LocalDevToolsAutoConfiguration {
return watcher;
}
+ @Configuration
+ @ConditionalOnBean(name = "sessionRedisTemplate")
+ static class RedisRestartConfiguration {
+
+ @Bean
+ public RestartCompatibleRedisSerializerConfigurer restartCompatibleRedisSerializerConfigurer(
+ RedisTemplate sessionRedisTemplate) {
+ return new RestartCompatibleRedisSerializerConfigurer(
+ sessionRedisTemplate);
+ }
+
+ }
+
}
}
diff --git a/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/RestartCompatibleRedisSerializerConfigurer.java b/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/RestartCompatibleRedisSerializerConfigurer.java
new file mode 100644
index 00000000000..7640a0c2781
--- /dev/null
+++ b/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/RestartCompatibleRedisSerializerConfigurer.java
@@ -0,0 +1,110 @@
+/*
+ * Copyright 2012-2015 the original author or authors.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.springframework.boot.devtools.autoconfigure;
+
+import javax.annotation.PostConstruct;
+
+import org.springframework.beans.factory.BeanClassLoaderAware;
+import org.springframework.core.convert.converter.Converter;
+import org.springframework.core.serializer.DefaultDeserializer;
+import org.springframework.core.serializer.support.DeserializingConverter;
+import org.springframework.core.serializer.support.SerializingConverter;
+import org.springframework.data.redis.core.RedisTemplate;
+import org.springframework.data.redis.serializer.RedisSerializer;
+import org.springframework.data.redis.serializer.SerializationException;
+import org.springframework.session.ExpiringSession;
+import org.springframework.util.ObjectUtils;
+
+/**
+ * Configures a {@link RedisTemplate} with a serializer for keys, values, hash keys, and
+ * hash values that is compatible with the split classloader used for restarts.
+ *
+ * @author Andy Wilkinson
+ * @author Rob Winch
+ *
+ * @see RedisTemplate#setHashKeySerializer(RedisSerializer)
+ * @see RedisTemplate#setHashValueSerializer(RedisSerializer)
+ * @see RedisTemplate#setKeySerializer(RedisSerializer)
+ * @see RedisTemplate#setValueSerializer(RedisSerializer)
+ */
+class RestartCompatibleRedisSerializerConfigurer implements BeanClassLoaderAware {
+
+ private final RedisTemplate redisTemplate;
+
+ private volatile ClassLoader classLoader;
+
+ RestartCompatibleRedisSerializerConfigurer(
+ RedisTemplate redisTemplate) {
+ this.redisTemplate = redisTemplate;
+ }
+
+ @Override
+ public void setBeanClassLoader(ClassLoader classLoader) {
+ this.classLoader = classLoader;
+ }
+
+ @PostConstruct
+ void configureTemplateSerializers() {
+ RestartCompatibleRedisSerializer serializer = new RestartCompatibleRedisSerializer(
+ this.classLoader);
+ this.redisTemplate.setHashKeySerializer(serializer);
+ this.redisTemplate.setHashValueSerializer(serializer);
+ this.redisTemplate.setKeySerializer(serializer);
+ this.redisTemplate.setValueSerializer(serializer);
+ }
+
+ static class RestartCompatibleRedisSerializer implements RedisSerializer