OpenFire源码学习之二十九:openfire集群配置

集群

Openfire的给集群提供了多种方案。一种是基于Hazelcast插件,还有基于Oraclecoherence插件。

Oraclecoherence插件中文开发文档:http://download.csdn.net/detail/huwenfeng_2011/8423519

Linux集群配置

一、修改配置文件

/etc/hosts文件

openfire1 192.168.2.104

openfire2192.168.2.240 

每台主机都需要配置

二、添加jar

coherence.jarcoherence-work.jartangosol.jar添加到lib目录

clustering.jar放到plugins下面

Hazelcast Clustering Plugin

Hazelcast插件添加支持运行多个冗余的Openfire服务器集群中的一起。作为群集运行的Openfire,您可以分发的几个服务器之间的连接负载,同时还提供故障转移服务器发生故障的事件。这个插件是更换为原来的Openfire收费集群插件,使用开源的Hazelcast数据分布框架代替昂贵的专有第三方产品。

安装

创建openfire集群,应该有至少两个的Openfire服务器,每个服务器必须安装Hazelcast插件。要安装Hazelcast ,只需hazelcast.jar拖放到 $OPENFIRE_HOME/plugins目录下,和其他安装的插件一样。您也可以使用插件的页面,从管理控制台安装插件。需要注意的是必须配置一个集群中的所有服务器共享一个单一的外部数据库(而不是嵌入式DB ) 。

Openfire的启动/初始化过程在默认情况下,服务器会发现对方UDP (组播)通过一个可配置的IP地址和端口的数据包交换。但是,请注意,许多其他初始化选项是可用的,并且可以使用,如果你的网络不支持组播通信(见下面的配置) 。

Hazelcast插件后已部署到每个服务器,在控制管理台上选择系统管理找到clustering界面,点击单选按钮。你只需要一次启用集群;这项更改会自动传播到其他服务器。刷新clustering页后,你将能看到所有服务器已成功加入集群。

需要注意的是Hazelcast与早期的clustering的插件(clustering.jar and enterprise.jar)是相互排斥的。 Hazelcast安装到您的Openfire服务器(S)之前,你将需要删除的任何现有旧的clutering 插件。

随着群集的建立和运行,现在你将需要某种形式的负载平衡器您的Openfire集群成员之间分发连接负载。有一些商业和开源替代品,例如,如果您使用的是HTTP/BOSH Openfire的连接器连接到OpenfireApache Web服务器(httpd)加上相应的代理平衡器模块(mod_proxy_balancer)可以提供一个可行的解决方案。其他一些流行的选择包括F5 LTM (商业)和HAProxy的(开源)等等。

一个简单的DNS轮循配置可以帮助分发XMPP连接跨多个Openfire的服务器集群中的。虽然作为一个轻量级的和低成本的方式提供基本的可扩展性,注意,这种做法是不考虑足够的真正的负载平衡,也不提供高可用性(HA) ,从客户的角度受欢迎。如果您正在评估这些选项,你可以在这里阅读更多。

Hazelcast插件升级

Hazelcast插件升级比较复杂,因为集群中的其他主机还依附着以前较老的版本。在openfire其中所有的主机应该都使用同一个版本的Hazelecast插件,以免出来各种错误或者数据同步问题。

选项1offline

注: 这个升级过程比较纯粹不会影响其他业务、数据,但是他需要产生暂短的服务终止

1、关闭openfire集群上的所有服务器

2、每台服务器都应该做如下操作

a) 删除先有插件hazelcast.jar 

b) 删除插件目录 plugins/hazelcasr

c) 将新的hazelcasr.jar插件复制到 plugins目录

d) 加载插件,重启openfire

3、在集群的其他插件重复2的操作

选项2online

注: 这个升级方法,openfire能够继续xmpp服务

1、在openfire集群上留一主机不需要关闭,关闭其他的主机

2、重复选项1的操作。

选项3Split-Brain

注: 如果你能访问到openfire控制台,则使用这种方法。如果他们连接到不同的服务器,  在升级过程中,用户之间可能不会通信。

 1从clusering页面上Openfire管理控制台,禁用集群。这将禁用集群的集群的所有成员。


    对于每个服务器,更新Hazelcast插件使用插件页面。
 2升级后的所有服务器上的插件,使用聚类页启用集群。这将激活集群集群的所有成员。

配置

Hazelcast插件内置到Openfire的系统属性有几个配置选项:

1、 hazelcast.startup.delay.seconds5):等待的秒数发射前Hazelcast插件。这允许的 Openfire之前部署任何其他插件初始化集群缓存等。

2、 hazelcast.startup.retry.count1):重试次数如果集群初始化启动失败,在第一次尝试。

3、 hazelcast.startup.retry.seconds10):等待后续尝试启动群集之间的秒数。

4、 hazelcast.max.execution.seconds30):当运行一个集群成员之间的同步任务的最长等待 时间。

5、 hazelcast.config.xml.filenamehazelcast缓存-config.xml)中:Hazelcast配置文件的名 称。通过覆盖这个值,你可以很容易地Hazelcast插件//目录中安装自定义缓存配置文 件,在命名的目录通过hazelcast.config.xml.directory的属性(如下所述),或在classpath 中您自己的自定义插件。

6、 hazelcast.config.xml.directory{OPENFIRE_HOME}/ conf目录下):目录将被添加到该 插件的类路径中。这使得Hazelcast一个自定义配置文件来Openfire的主目录之外。

7、 hazelcast.config.jmx.enabledfalse):Hazelcast集群如果已启用JMX通过Openfire管理 控制台启用JMX支持。有关更多信息,请参阅Hazelcast JMX文档。...

Hazelcast插件使用XML配置生成器,从XML文件中上述初始化集群。默认情况下,集群成员试图通过多播发现对方在以下位置:

IP地址:224.2.2.3
    端口:54327

注意,这些值可以被覆盖的plugin's /classes/hazelcast-cache-config.xml(通过组播组和组播端口元素)。许多其它的初始化和发现存在Hazelcast配置文档如上所述记录。例如,成立一个双节点群集使用众所周知的DNS名称/端口值,请尝试以下选择:

[html]  view plain  copy
  1. <network>  
  2.     <port auto-increment="true">5701</port>  
  3.     <join>  
  4.         <multicast enabled="false"/>  
  5.         <tcp-ip enabled="true">  
  6.             <hostname>192.169.1.240:5701</hostname>  
  7.             <hostname>192.169.1.250:5701</hostname>  
  8. <hostname>192.169.1.120:5701</hostname>  
  9. <hostname>192.169.1.104:5701</hostname>  
  10.         </tcp-ip>  
  11.         <aws enabled="false"/>  
  12. </join>  
  13.     <interfaces enabled="false"/>  
  14.     <ssl enabled="false" />  
  15.     <socket-interceptor enabled="false" />  
  16.     <symmetric-encryption enabled="false">  
  17.         <!--  
  18.            encryption algorithm such as  
  19.            DES/ECB/PKCS5Padding,  
  20.            PBEWithMD5AndDES,  
  21.            AES/CBC/PKCS5Padding,  
  22.            Blowfish,  
  23.            DESede  
  24.         -->  
  25.         <algorithm>PBEWithMD5AndDES</algorithm>  
  26.         <!-- salt value to use when generating the secret key -->  
  27.         <salt>thesalt</salt>  
  28.         <!-- pass phrase to use when generating the secret key -->  
  29.         <password>thepass</password>  
  30.         <!-- iteration count to use when generating the secret key -->  
  31.         <iteration-count>19</iteration-count>  
  32.     </symmetric-encryption>  
  33.     <asymmetric-encryption enabled="false">  
  34.         <!-- encryption algorithm -->  
  35.         <algorithm>RSA/NONE/PKCS1PADDING</algorithm>  
  36.         <!-- private key password -->  
  37.         <keyPassword>thekeypass</keyPassword>  
  38.         <!-- private key alias -->  
  39.         <keyAlias>local</keyAlias>  
  40.         <!-- key store type -->  
  41.         <storeType>JKS</storeType>  
  42.         <!-- key store password -->  
  43.         <storePassword>thestorepass</storePassword>  
  44.         <!-- path to the key store -->  
  45.         <storePath>keystore</storePath>  
  46.     </asymmetric-encryption>  
  47. </network>  

进入控制台管理页面:

在这里选择启动,并保存设置。然后等待几分钟,会出现集群内的主机


接下来点击一个节点,查看里面的内容

源码分析

openfire的核心原理集成的缓存插件CacheFactory中方法:

isClusteringAvailable()

该方法返回true,如果群集安装,可以使用此JVM加入集群。 false值可能意味着,要么集群支持是不可用或许可证不允许有超过1群集节点。

[java]  view plain  copy
  1. public static boolean isClusteringAvailable() {  
  2.         if (clusteredCacheFactoryStrategy == null) {  
  3.             try {  
  4.                 clusteredCacheFactoryStrategy = (CacheFactoryStrategy) Class.forName(  
  5.                         clusteredCacheFactoryClass, true,  
  6.                         getClusteredCacheStrategyClassLoader()).newInstance();  
  7.             } catch (Exception e) {  
  8.                 log.warn("Clustered cache factory strategy " + clusteredCacheFactoryClass + " not found");  
  9.             }  
  10.         }  
  11.         return (clusteredCacheFactoryStrategy != null);  
  12.     }  

加入集群的时候会先判断该主机是否已经加入集群,这里就是要判断该主机是否加入了集群,如果加入集群的话则会重载分布式缓存。

[java]  view plain  copy
  1. public static synchronized void startup() {  
  2.         if (isClusteringEnabled() && !isClusteringStarted()) {  
  3.             initEventDispatcher();  
  4.             CacheFactory.startClustering();  
  5.         }  
  6.     }  

上面方法中有两个步骤:

1)initEventDispatcher();调度线程实例化和启动群集事件

2)CacheFactory.startClustering();启动集群

[java]  view plain  copy
  1. private static void initEventDispatcher() {  
  2.         if (dispatcher == null || !dispatcher.isAlive()) {  
  3.             dispatcher = new Thread("ClusterManager events dispatcher") {  
  4.                 @Override  
  5.                 public void run() {  
  6.                     // 当集群被禁止,退出线程  
  7.                     while (ClusterManager.isClusteringEnabled()) {  
  8.                         try {  
  9.                             Event event = events.take();  
  10.                             EventType eventType = event.getType();  
  11.                             // 更新缓存的时候,让CacheFacory最先得到这个事件  
  12.                             if (event.getNodeID() == null) {  
  13.                                 // Replace standalone caches with clustered caches and migrate data  
  14.                                 if (eventType == EventType.joined_cluster) {  
  15.                                     CacheFactory.joinedCluster();  
  16.                                 } else if (eventType == EventType.left_cluster) {  
  17.                                     CacheFactory.leftCluster();  
  18.                                 }  
  19.                             }  
  20.                             // Now notify rest of the listeners  
  21.                             for (ClusterEventListener listener : listeners) {  
  22.                                 try {  
  23.                                     switch (eventType) {  
  24.                                         case joined_cluster: {  
  25.                                             if (event.getNodeID() == null) {  
  26.                                                 listener.joinedCluster();  
  27.                                             }  
  28.                                             else {  
  29.                                                 listener.joinedCluster(event.getNodeID());  
  30.                                             }  
  31.                                             break;  
  32.                                         }  
  33.                                         case left_cluster: {  
  34.                                             if (event.getNodeID() == null) {  
  35.                                                 listener.leftCluster();  
  36.                                             }  
  37.                                             else {  
  38.                                                 listener.leftCluster(event.getNodeID());  
  39.                                             }  
  40.                                             break;  
  41.                                         }  
  42.                                         case marked_senior_cluster_member: {  
  43.                                             listener.markedAsSeniorClusterMember();  
  44.                                             break;  
  45.                                         }  
  46.                                         default:  
  47.                                             break;  
  48.                                     }  
  49.                                 }  
  50.                                 catch (Exception e) {  
  51.                                     Log.error(e.getMessage(), e);  
  52.                                 }  
  53.                             }  
  54.                             // Mark event as processed  
  55.                             event.setProcessed(true);  
  56.                         } catch (Exception e) {  
  57.                             Log.warn(e.getMessage(), e);  
  58.                         }  
  59.                     }  
  60.                 }  
  61.             };  
  62.             dispatcher.setDaemon(true);  
  63.             dispatcher.start();  
  64.         }  
  65.     }  
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值