Spring   发布时间:2022-04-09  发布网站:大佬教程  code.js-code.com
大佬教程收集整理的这篇文章主要介绍了如何使用嵌入式tomcat会话群集设置Spring Boot应用程序?大佬教程大佬觉得挺不错的,现在分享给大家,也给大家做个参考。

我想设置一个带有嵌入式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不共享?

便说一句,我读过< distributed="">必须在web.xml上指定tag才能使应用程序使用tomcat会话群集.但我不知道如何使用Spring Boot的no-xml环境来指定它.这是问题的原因吗?然后如何指定?

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

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

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

@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.

我有点担心这个版本依赖.所以我opened an issue为此.

大佬总结

以上是大佬教程为你收集整理的如何使用嵌入式tomcat会话群集设置Spring Boot应用程序?全部内容,希望文章能够帮你解决如何使用嵌入式tomcat会话群集设置Spring Boot应用程序?所遇到的程序开发问题。

如果觉得大佬教程网站内容还不错,欢迎将大佬教程推荐给程序员好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。