我们实现了由Redis支持的Spring会话,并有一个Tomcat服务器集群。当我们通过不设置jvmRoute关闭粘滞会话时,我们一直在jcaptcha服务中获得“文本验证失败”。我认为这是因为jcaptcha servlet对servlet一无所知,它拥有所有Spring会话过滤器,因此无法读取会话变量。我们如何使jcaptcha与Spring会话一起工作?
这是我们的设置:
Web.xml
<servlet>
<servlet-name>my-servlet</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>throwExceptionIfNoHandlerFound</param-name>
<param-value>true</param-value>
</init-param>
<load-on-startup>2</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>my-servlet</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>jcaptcha</servlet-name>
<servlet-class>com.octo.captcha.module.servlet.image.SimpleImageCaptchaServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>jcaptcha</servlet-name>
<url-pattern>/jcaptcha/jcaptcha.jpg</url-pattern>
</servlet-mapping>CustomHttpSessionAppInitializer.java
public class CustomHttpSessionAppInitializer extends AbstractHttpSessionApplicationInitializer {}RedisSessionConfig.java
@Configuration
@EnableRedisHttpSession
public class RedisSessionConfig {
@Value("${spring.redis.host}")
private String redisServerName;
@Value("${spring.redis.port}")
private Integer redisServerPort;
@Value("${spring.redis.database}")
private Integer redisServerDatabase;
@Value("${spring.redis.password}")
private String redisServerPassword;
@Value("${spring.server.affinity}")
private Boolean isServerAffinity = Boolean.FALSE;
@Autowired
private SessionIdentifierService sessionIdentifierService;
@Bean
public JedisConnectionFactory jedisConnectionFactory() {
RedisStandaloneConfiguration config = new RedisStandaloneConfiguration(redisServerName, redisServerPort);
config.setDatabase(redisServerDatabase);
config.setPassword(RedisPassword.of(redisServerPassword));
return new JedisConnectionFactory(config);
}
/*
* We need to register every HttpSessionListener as a bean to translate SessionDestroyedEvent and SessionCreatedEvent into
* HttpSessionEvent. Otherwise we will got a lot of warning messages about being Unable to publish Events for the session.
* See Spring Session Docs at:
* {@link} https://docs.spring.io/spring-session/docs/current/reference/html5/#httpsession-httpsessionlistener
*/
@Bean
public HttpSessionEventPublisher httpSessionEventPublisher() {
return new HttpSessionEventPublisher();
}
@Bean
public CookieSerializer cookieSerializer() {
DefaultCookieSerializer serializer = new DefaultCookieSerializer();
serializer.setCookieName("JSESSIONID");
serializer.setUseBase64Encoding(false);
if (isServerAffinity) {
serializer.setJvmRoute(sessionIdentifierService.getJvmRoute());
}
return serializer;
}
@Bean
public RedisTemplate<String, Object> redisTemplate() {
RedisTemplate<String, Object> template = new RedisTemplate<>();
template.setConnectionFactory(jedisConnectionFactory());
return template;
}
}发布于 2019-02-07 23:14:39
将jcaptcha与Spring会话集成应该没有问题。只要有从Redis加载会话的方法(在本例中是通过会话cookie ),并且会话存在,调用request.getSession()或request.getSession(false)就会返回Redis支持的会话。
这可以在任何称为的过滤器和servlet中工作,这些过滤器和servlet的名称是 springSessionRepositoryFilter。如果您查看SessionRepositoryFilter的源代码,您将看到HttpServletRequest与SessionRepositoryRequestWrapper交换。
因此,您的SimpleImageCaptchaServlet和用于验证用户响应的servlet都将获得一个SessionRepositoryRequestWrapper,该SessionRepositoryRequestWrapper似乎将允许您访问Redis支持的会话。
问题可能是您的配置;springSessionRepositoryFilter可能没有在容器中注册,特别是因为您同时使用web.xml和Servlet 3.0+ WebApplicationInitializer。如果您的应用程序工作正常,那么您的web.xml很可能运行良好。您是否使用WebApplicationInitializer加载您的web.xml?如果不是,那么可能是您的Java没有加载。确保您的web.xml以某种方式加载您的配置,可能是通过启用Java文件中的组件扫描(<context:component-scan/>)来加载您的Java以及:
<bean class="org.springframework.session.data.redis.config.annotation.web.http.RedisHttpSessionConfiguration"/>要加载将创建筛选器的配置,必须将其添加到web.xml中:
<filter>
<filter-name>springSessionRepositoryFilter</filter-name>
<filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
</filter>
<filter-mapping>
<filter-name>springSessionRepositoryFilter</filter-name>
<url-pattern>/*</url-pattern>
<dispatcher>REQUEST</dispatcher>
<dispatcher>ERROR</dispatcher>
</filter-mapping>https://stackoverflow.com/questions/54046802
复制相似问题