我希望能设置一个集成了Tomcat Session Cluster的Spring Boot应用程序。
由于嵌入式Tomcat没有
显然,HttpSession没有得到共享。但是当第二个实例创建一个新会话时,第一个实例的登录信息被清除并且登录无效。
这里发生了什么?Session被共享但HttpSession没有被共享?
顺便说一下,我已经知道标签必须在web.xml上进行指定,以便应用程序使用Tomcat会话集群。但是我不知道如何在Spring Boot的无XML环境中指定它。这是问题的原因吗?那么该如何指定呢?
我已经搜索并找到了一些文档,展示了使用Redis进行集群。但是目前我不想在我的配置中添加另一个移动部件。在我的配置中,3~4个节点是最大的。
由于嵌入式Tomcat没有
server.xml
文件,因此我创建了一个TomcatEmbeddedServletContainerFactory并以编程方式设置集群配置。代码如下:@Configuration
public class TomcatConfig
{
@Bean
public EmbeddedServletContainerFactory servletContainerFactory()
{
return new TomcatEmbeddedServletContainerFactory()
{
@Override
protected TomcatEmbeddedServletContainer getTomcatEmbeddedServletContainer(
Tomcat tomcat)
{
configureCluster(tomcat);
return super.getTomcatEmbeddedServletContainer(tomcat);
}
private void configureCluster(Tomcat tomcat)
{
// static membership cluster
SimpleTcpCluster cluster = new SimpleTcpCluster();
cluster.setChannelStartOptions(3);
{
DeltaManager manager = new DeltaManager();
manager.setNotifyListenersOnReplication(true);
cluster.setManagerTemplate(manager);
}
{
GroupChannel channel = new GroupChannel();
{
NioReceiver receiver = new NioReceiver();
receiver.setPort(localClusterMemberPort);
channel.setChannelReceiver(receiver);
}
{
ReplicationTransmitter sender = new ReplicationTransmitter();
sender.setTransport(new PooledParallelSender());
channel.setChannelSender(sender);
}
channel.addInterceptor(new TcpPingInterceptor());
channel.addInterceptor(new TcpFailureDetector());
channel.addInterceptor(new MessageDispatch15Interceptor());
{
StaticMembershipInterceptor membership =
new StaticMembershipInterceptor();
String[] memberSpecs = clusterMembers.split(",", -1);
for (String spec : memberSpecs)
{
ClusterMemberDesc memberDesc = new ClusterMemberDesc(spec);
StaticMember member = new StaticMember();
member.setHost(memberDesc.address);
member.setPort(memberDesc.port);
member.setDomain("MyWebAppDomain");
member.setUniqueId(memberDesc.uniqueId);
membership.addStaticMember(member);
}
channel.addInterceptor(membership);
}
cluster.setChannel(channel);
}
cluster.addValve(new ReplicationValve());
cluster.addValve(new JvmRouteBinderValve());
cluster.addClusterListener(new ClusterSessionListener());
tomcat.getEngine().setCluster(cluster);
}
};
}
private static class ClusterMemberDesc
{
public String address;
public int port;
public String uniqueId;
public ClusterMemberDesc(String spec) throws IllegalArgumentException
{
String[] values = spec.split(":", -1);
if (values.length != 3)
throw new IllegalArgumentException("clusterMembers element " +
"format must be address:port:uniqueIndex");
address = values[0];
port = Integer.parseInt(values[1]);
int index = Integer.parseInt(values[2]);
if ((index < 0) || (index > 255))
throw new IllegalArgumentException("invalid unique index: must be >= 0 and < 256");
uniqueId = "{";
for (int i = 0; i < 16; i++, index++)
{
if (i != 0)
uniqueId += ',';
uniqueId += index % 256;
}
uniqueId += '}';
}
};
// This is for example. In fact these are read from application.properties
private int localClusterMemberPort = 9991;
private String clusterMembers = "111.222.333.444:9992:1";
}
我使用以下环境测试了代码:
- 单个Windows电脑
- 2个Spring Boot应用程序实例,具有不同的localClusterMemberPort和clusterMembers
由于cookie不考虑端口,因此包含JSESSIONID的cookie在两个实例之间共享。
当启动实例时,Tomcat集群似乎可以工作,因为2个实例请求的JSESSIONID值相同。但是,在使用第一个实例登录后向第二个实例发出请求时,第二个实例无法找到HttpSession,并记录以下消息:
w.c.HttpSessionSecurityContextRepository : No HttpSession currently exists
w.c.HttpSessionSecurityContextRepository : No SecurityContext was available from the HttpSession: null. A new one will be created.
显然,HttpSession没有得到共享。但是当第二个实例创建一个新会话时,第一个实例的登录信息被清除并且登录无效。
这里发生了什么?Session被共享但HttpSession没有被共享?
顺便说一下,我已经知道标签必须在web.xml上进行指定,以便应用程序使用Tomcat会话集群。但是我不知道如何在Spring Boot的无XML环境中指定它。这是问题的原因吗?那么该如何指定呢?
我已经搜索并找到了一些文档,展示了使用Redis进行集群。但是目前我不想在我的配置中添加另一个移动部件。在我的配置中,3~4个节点是最大的。