jboss集群session复制原理:
jboss session复制是jboss session同步的一种实现。原理是在各Jboss节点间建立横向联系,每个节点都将本节点的session变化同步到其他所有节点上。
jboss的session复制与HTTP集群是相互配合、相互独立的两个系统。session复制是节点间的横向联系,HTTP集群是负载均衡器与节点的纵向联系。
10.0.2.203 – 安装JDK 1.6,JBoss 7,JBoss节点名称为jboss1
10.0.2.202 – 安装JDK 1.6,JBoss 7,JBoss节点名称为jboss2
10.0.2.200 – 安装Apache httpd,mod_jk #需要在官网下载mod_jk模块,重命名为mod_jk,放置于/etc/httpd/modules 下
以下实验注意iptables 和 selinux 的影响。
安装jdk和jboss 此处就不多说了,jboss不需要修改
/usr/local/jboss-4.2.2.GA/server/all/deploy/jboss-web.deployer 目录下的
server.xml文件的 <Connector port="8080" address="${jboss.bind.address}"默认地址,使用 mod_proxy时修改了此处。
下面是apache 的配置:
a:在/etc/httpd/conf.d/目录下创建mod_jk.conf文件,内容为:
LoadModule jk_module modules/mod_jk.so #指定模块路径
JkWorkersFile conf.d/workers.properties #指定mod_jk的workers定义
JkLogFile logs/mod_jk.log #指定jk的日志输出文件
JkLogLevel info #指定日志级别
JkLogStampFormat "[%a %b %d %H:%M:%S: %Y]" #用来配置log文件的日期/时间格式. 使用strftime()的格式化字符串,默认是[%a %b %d %H:%M:%S %Y]
#Options Description(选项的说明)
# %b 发送的字节, 不包括 HTTP headers (CLF format)
# %B 发送的字节, 不包括 HTTP headers
# %H 协议
# %m 请求方式(get/post)
# %p 服务器响应请求的规范端口.
# %q 查询字符串 (如果存在以?开头,否则是空串)
# %r 请求的第一行.
# %s HTTP状态码
# %T 请求间隔, 处理请求耗费的时间 秒.微秒
# %U 请求的url路径,不包含查询字符串.
# %v 响应请求的规范服务器名字
# %V 根据UseCanonicalName设置的服务器名字.
# %w Tomcat worker 名字
JkOptions +ForwardKeySize +ForwardURICompat -ForwardDirectories
JkRequestLogFormat "%w %V %T" #设置个人用户请求的log格式
#JkMount /* loadbalancer
JkMountFile conf.d/uriworkermap.properties #加载workers的请求处理分配文件
#JkMount /* status
JkShmFile logs/jk.shm #共享内存
<Location /jkstatus/> 对/jkstatus 新建访问控制
JkMount status
Order deny,allow
Deny from all
Allow from 10.0.2.0
</Location>
b:接着在/etc/httpd/conf.d/目录下创建workers.properties 文件,内容为:
worker.list=loadbalancer,status # worker列表
worker.jboss1.port=8009 #ajp13 端口号
worker.jboss1.host=10.0.2.203
worker.jboss1.type=ajp13
worker.jboss1.lbfactor=1 #server的加权比重,值越高,分得的请求越多
worker.jboss1.cachesize=10
worker.jboss2.port=8009
worker.jboss2.host=10.0.2.202
worker.jboss2.type=ajp13
worker.jboss2.lbfactor=1
worker.jboss2.cachesize=10
#重试次数
worker.retries=3
worker.loadbalancer.type=lb #用于负载均衡
worker.loadbalancer.balance_workers=jboss1,jboss2 #指定分担请求的server列表,用逗号分隔
worker.loadbalancer.sticky_session=1 #设置用于负载均衡的server的session可否共享
#一次请求中 session 只会在一个jboss上,当该jboss奔溃后,请求会跳转到其他jboss上,但是其session不会复制到跳转后的jboss上
worker.loadbalancer.sticky_session_force=0 #当jboss1崩溃后,请求转发到jboss2
worker.status.type=status #描述是用于httpd自身状态监控的status
c:然后 在/etc/httpd/conf.d/目录下创建uriworkermap.properties 文件,内容为:
/*=loadbalancer #所有请求都由loadbalancer这个server处理
/jkstatus=status #所有包含jkstatus请求的都由status这个server处理
!/*.gif=loadbalancer #所有以.gif结尾的请求都不由loadbalancer这个server处理,以下几个都是一样的意思
!/*.jpg=loadbalancer #这里的"!”类似于java中的"!”,是“非”的意思
!/*.png=loadbalancer
!/*.css=loadbalancer
!/*.js=loadbalancer
!/*.htm=loadbalancer
!/*.html=loadbalancer
以下是jboss 实例的配置:
a:在10.0.2.203主机,编辑 $JBOSS_HOME/server/all/deploy/jboss-web.deployer/server.xml 文件
<Engine name="jboss.web" defaultHost="localhost" jvmRoute="jboss1">
在Engine 处添加 jvmRoute="jboss1",在10.0.2.202主机也相应地修改为jboss2
b:在10.0.2.203和10.0.2.202主机分别编辑 $JBOSS_HOME/server/all/deploy/jboss-web.deployer/
META-INF/jboss-service.xml 文件
<attribute name="UseJK">true</attribute> #将UseJK的值改为true,使用mod_jk
c:在10.0.2.203和10.0.2.202主机分别编辑
/usr/local/jboss-4.2.2.GA/server/all/deploy/jboss-web.deployer/conf/web.xml 文件
<?xml version="1.0"?>
<web-app xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
version="2.4">
<distributable/> #添加此项,开启jboss的session复制
<!-- ... -->
</web-app>
d:这里要做的是基于request的cluster,也就让各个节点之间互相复制session状态。有两种复制模式,同步与异步。使用同步的方式,jboss会把session复制的操作和对request的响应放到一个应用事务(application transaction),session复制完成后才去处理request。异步复制则发送session复制的消息后马上处理request,session复制则会稍有延迟。但是在多框架的web页面中,这样的集群方式会有问题。由于frame在同一时间发出多个request,会造成一些混乱,这也是采用基于用户的集群方式的原因之一。
<attribute name="CacheMode">REPL_ASYNC</attribute>
REPL_ASYNC(异步)或者REPL_SYNC(同步)。
attribute name="ClusterName"> 这个属性是设置你的集群的名字
在下面还有个<attribute name="ClusterConfig"> 这个属性是设置每个节点在进行session复制的时候是如何通信的,有TCP和UDP两种,默认使用UDP。如果使用udp方式,那么应该将udp的lookback属性指定为true。采用UDP方式问题比较多,所以采用TCP方式,应该指定bind_addr的值为本机ip,并且在TCPPING标签的initial_hosts属性中列出所有节点,格式是”机器名[端口号]”,将<TCP>到</TCP>标签内的全部down_thread和up_thread的false都改为true。还需要注释掉<!--<config><udp>...</udp></config>-->。
e:在10.0.2.203和10.0.2.202 主机写index.jsp测试页面,放置于/usr/local/jboss-4.2.2.GA/server/all/deploy/jboss-web.deployer/ROOT.war 目录下,内容为:
<%@ page contentType="text/html; charset=GBK" %>
<%@ page import="java.util.*" %>
<html><head><title>Cluster App Test</title></head>
<body>
Server Info:
<%
out.println(request.getLocalAddr() + " : " + request.getLocalPort()+"<br>");%>
<%
out.println("<br> ID " + session.getId()+"<br>");
String dataName = request.getParameter("dataName");
if (dataName != null && dataName.length() > 0) {
String dataValue = request.getParameter("dataValue");
session.setAttribute(dataName, dataValue);
}
out.print("<b>Session list</b>");
Enumeration e = session.getAttributeNames();
while (e.hasMoreElements()) {
String name = (String)e.nextElement();
String value = session.getAttribute(name).toString();
out.println( name + " = " + value+"<br>");
System.out.println( name + " = " + value);
}
%>
%<form action="index.jsp" method="POST">
%id:<input type=text size=20 name="dataName">
%<br>
%key:<input type=text size=20 name="dataValue">
%<br>
%<input type=submit>
%</form>
%</body>
%</html>
最后浏览器访问:10.0.2.90/jkstatus
可以从jkstatus看到关于设置的大量信息
浏览器访问10.0.2.90/index.jsp
接着填写内容并提交
可以看到提交的内容并sticky在一台jboss实例
此时将jboss1停掉,可以看到session在jboss2上面,提交内容,再启动jboss1,停掉jboss2,可以看到session在jboss1上面。(此实验session从jboss2到jboss1有点问题,待解决)
查看/usr/local/jboss-4.2.2.GA/server/all/deploy/jboss-web-cluster.sar/META-INF目录下的jboss-web.xml 文件:
ClusterName是集群名称,
在同一局域网内,可以存在多个jboss集群,根据集群名称区分它们.所以,集群中各节点配置的集群名称必须一致,而机器IP则没有特殊要求,只要它们能相互连通. 理论上,可以在一台机器上安装多个Jboss实例,分属于不同的集群.但这会极大地增加复杂度,是不好的配置方式.IsolationLevel是隔离等级.
可选值包括:SERIALIZABLE, REPEATABLE_READ, READ_COMMITTED, READ_UNCOMMITTED, 和 NONE。这里的隔离级别和数据库的隔离级别有同样的含义,对于大多数WEB应用程序来讲通常设置为REPEATABLE_READ。CacheMode是缓存模式。
由于session复制是通过缓存实现的,所以实际上是复制模式.可选值包括:REPL_SYNC 和REPL_ASYNC,确定改变是应该同步还是异步复制。缺省值是REPL_ASYNC.使用同步复制,确保在请求完成之前传播改变,session同步没有滞后,但效率低。