首页 文章

如何使用嵌入式tomcat会话群集设置Spring Boot应用程序?

提问于
浏览
7

我想设置一个带有嵌入式tomcat会话群集的Spring Boot应用程序 .

由于嵌入式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 PC

  • 2具有不同localClusterMemberPort和clusterMembers的Spring启动应用程序实例

由于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没有被共享 . 但是,当第二个实例创建新会话时,将清除第一个实例的登录信息并使登录失效 .

这里发生了什么?会话是共享但HttpSession不共享?

顺便说一下,我已经读过必须在 web.xml 上指定 <distributed /> 标签才能让应用程序使用tomcat会话群集 . 但是我没有_289117_的no-xml环境 . 这是问题的原因吗?然后如何指定?

我搜索过并发现了一些使用Redis显示聚类的文档 . 但目前我不想在我的配置中添加另一个移动部件 . 在我的配置中,3~4个节点是最大的 .

2 回答

  • 0

    关键是使上下文可分配,并设置管理器 .

    当我按如下方式修改问题的代码时,会话群集起作用 .

    @Configuration
    public class TomcatConfig
    {
        @Bean
        public EmbeddedServletContainerFactory servletContainerFactory()
        {
            TomcatEmbeddedServletContainerFactory factory = new TomcatEmbeddedServletContainerFactory()
            {
                ...
            };
    
            factory.addContextCustomizers(new TomcatContextCustomizer()
            {
                @Override
                public void customize(Context context)
                {
                    context.setManager(new DeltaManager());
                    context.setDistributable(true);
                }
            });
    
            return factory;
        }
    
        ...
    }
    

    对于Spring Boot 1.2.4,不需要context.setManager() . 但是对于Spring Boot到1.3.0,如果未调用context.setManager(),则群集将失败并显示以下日志 .

    2015-11-18 19:59:42.882  WARN 9764 --- [ost-startStop-1] o.a.catalina.ha.tcp.SimpleTcpCluster     : Manager [org.apache.catalina.session.StandardManager[]] does not implement ClusterManager, addition to cluster has been aborted.
    

    我有点担心这个版本依赖 . 所以我就这样做了.2689120_ .

  • 7

    在Spring Boot 2.0.x中,您需要使用 WebServerFactoryCustomizer 来配置群集 .

    @Component
    public class WebServerCustomizer implements WebServerFactoryCustomizer<TomcatServletWebServerFactory> {
        @Override
        public void customize( final TomcatServletWebServerFactory factory ) {
            factory.addContextCustomizers( new TomcatClusterContextCustomizer() );
        }
    }
    
    public class TomcatClusterContextCustomizer implements TomcatContextCustomizer {
        @Override
        public void customize( final Context context ) {
            // Call method defined in the question text above, but pass Engine 
            // instead of Tomcat
            configureCluster( (Engine)context.getParent().getParent() );
        }
    }
    

相关问题